Python super()函数

发布于 2020-06-20  812 次阅读


描述

super() 函数是用于调用父类(超类)的一个方法(不用明确给出任何基类的名字,它会自动找到所有直接基类,及其方法用于继承)。

super 是用来解决多重继承问题的,直接用类名调用父类方法在使用单继承的时候没问题,但是如果使用多继承,会涉及到查找顺序(MRO)、重复调用(钻石继承)等种种问题。

MRO 就是类的方法解析顺序表, 其实也就是继承父类方法时的顺序表。

语法

以下是 super() 方法的语法:

super(type[, object-or-type])

参数

  • type -- 类。
  • object-or-type -- 类,一般是 self

Python3.x 和 Python2.x 的一个区别是: Python 3 可以使用直接使用 super().xxx 代替 super(Class, self).xxx

单继承

在单继承时,super().__init__()与Base().__init__()是一样的。super().__init__()避免了显式调用(有一些函数你主动调用了就是显式调用,如果你没有调用、但确实这个函数被调用了这就是隐式调用)

class Base:
    def __init__(self):
        print('Create Base')


class A(Base):
    def __init__(self):
        # Base.__init__(self)
        # super(A, self).__init__() # Python2
        super().__init__() # Python3
        print('Create A')

A()
##########
Create Base
Create A

多继承

super跟父类没有什么实际性的关联,在单继承时super获取的类刚好是父类,在多继承时,super获取的类继承顺序中的下一个类。如下例子

  Base
  /  \
 /    \
A      B
 \    /
  \  /
   C

使用super().__init__()方式如下

class Base:
    def __init__(self):
        print ("enter Base")
        print ("leave Base")

class A(Base):
    def __init__(self):
        print ("enter A")
        super().__init__()
        print ("leave A")

class B(Base):
    def __init__(self):
        print ("enter B")
        super().__init__()
        print ("leave B")

class C(A, B):
    def __init__(self):
        print ("enter C")
        super().__init__()
        print ("leave C")

C()
#############################
enter C
enter A
enter B
enter Base
leave Base
leave B
leave A
leave C

使用Base().__init__()

class Base:
    def __init__(self):
        print ("enter Base")
        print ("leave Base")

class A1(Base):
    def __init__(self):
        print ("enter A")
        Base().__init__()
        print ("leave A")

class B1(Base):
    def __init__(self):
        print ("enter B")
        Base().__init__()
        print ("leave B")

class C1(A1, B1):
    def __init__(self):
        print ("enter C")
        A1().__init__()
        B1().__init__()
        print ("leave C")

C1()
#############################
enter C
enter A
enter Base
leave Base
enter Base
leave Base
leave A
enter A
enter Base
leave Base
enter Base
leave Base
leave A
enter B
enter Base
leave Base
enter Base
leave Base
leave B
enter B
enter Base
leave Base
enter Base
leave Base
leave B
leave C
从上面看到,不使用super 基类会多次被调用,开销非常大
对于定义的类,在Python中会创建一个MRO(Method Resolution Order)列表,它代表了类继承的顺序。查看MRO列表:

使用super

class Base:
    def __init__(self):
        print ("enter Base")
        print ("leave Base")

class A(Base):
    def __init__(self):
        print ("enter A")
        super().__init__()
        print ("leave A")

class B(Base):
    def __init__(self):
        print ("enter B")
        super().__init__()
        print ("leave B")

class C(A, B):
    def __init__(self):
        print ("enter C")
        super().__init__()
        print ("leave C")

# C()
print(C.mro())
##########################
[<class '__main__.C'>, <class '__main__.A'>, <class '__main__.B'>, <class '__main__.Base'>, <class 'object'>]

不使用super

class A1(Base):
    def __init__(self):
        print ("enter A")
        Base().__init__()
        print ("leave A")

class B1(Base):
    def __init__(self):
        print ("enter B")
        Base().__init__()
        print ("leave B")

class C1(A1, B1):
    def __init__(self):
        print ("enter C")
        A1().__init__()
        B1().__init__()
        print ("leave C")

# C1()
print(C1.mro())
############################
[<class '__main__.C1'>, <class '__main__.A1'>, <class '__main__.B1'>, <class '__main__.Base'>, <class 'object'>]

总结

  • 从测试结果来看,两种方式的MRO列表是一样的。MRO的查找顺序(python3)是按广度优先来的(基类继承object,Python2是深度优先)
  • super的好处是避免直接使用父类的名字(隐式调用),主要用于多层继承
  • 避免使用 super(self.__class__, self)
  • 在super机制里可以保证公共父类仅执行一次,执行的顺序按照MRO(method resolution order)方法解析列表顺序来进行的


一名测试工作者,专注接口测试、自动化测试、性能测试、Python技术。