python基础知识
用于类继承的super函数介绍
目录
一、super函数的用途
(1)避免在改动父类名称时还需改动子类调用方法的代码
(2)在子类中按照一套内置的顺序自动调用父类的方法
(3)多用于多继承问题中,解决查找顺序(MRO)、重复调用(钻石继承)等种种问题
二、了解super函数的基本信息
super([type[, object-or-type]])
函数说明:
返回一个代理对象,它会将方法调用委托给 type 的父类或兄弟类。
参数说明 :
type:类,可选参数;
object-or-type:对象或类,一般是self,也是可选参数;
参数作用:第二个参数决定了调用父类的顺序,比如self,就是调用super函数的对象本身,它有一个__mro__属性,决定了按什么顺序调用父类的方法;第一个参数决定了从哪个类开始调用,一般也可以是本身类,即从本身这个类的顺序中的下一个类开始调用该方法
注:但要注意调用该方法的从始至终都是self对象,而不是父类对象
三、多继承不重复调用
这里我们就直接从多继承开始,明白了多继承super函数的使用,单继承自然也就不在话下。
代码如下:
class A:
def __init__(self):
print("找到 funxx() 位于 A 中...")
class B(A):
def __init__(self):
print("找到 funxx() 位于 B 中...")
class C(A):
def __init__(self):
print("找到 funxx() 位于 C 中...")
pass
class D(A):
def __init__(self):
print("找到 funxx() 位于 D 中...")
class E(B, C):
def __init__(self):
print("找到funcxx()位于E中...")
class F(E, D):
def __init__(self):
print("执行 F 中的 funff()...")
super(F, self).__init__()
print(f"F 类的 MRO : {F.__mro__}")#获取得到F类的__mro__属性,从而知道其调用父类的顺序
f = F()
输出如下:
F 类的 MRO : (<class '__main__.F'>, <class '__main__.E'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.D'>, <class '__main__.A'>, <class 'object'>)
执行 F 中的 funff()...
找到funcxx()位于E中...
(1)通过__mro__属性得到F类调用其父类的顺序;即输出的第一行,按照F->E->B->C->D->A类的顺序调用其__init__方法
(2)我们看到super函数内的第二个参数为self对象,所以其按照上述顺序进行调用父类方法;第一个参数为F,即F类,所以我们从F往下开始调用父类方法,这里也就是E类的方法
再次实践(这次我们将super函数的第一个参数改为B):
class A:
def __init__(self):
print("找到 funxx() 位于 A 中...")
class B(A):
def __init__(self):
print("找到 funxx() 位于 B 中...")
class C(A):
def __init__(self):
print("找到 funxx() 位于 C 中...")
pass
class D(A):
def __init__(self):
print("找到 funxx() 位于 D 中...")
class E(B, C):
def __init__(self):
print("找到funcxx()位于E中...")
class F(E, D):
def __init__(self):
print("执行 F 中的 funff()...")
super(B, self).__init__()
print(f"F 类的 MRO : {F.__mro__}")#获取得到F类的__mro__属性,从而知道其调用父类的顺序
f = F()
显然就应该调用该顺序中B类的下一个类C类的方法;输出如下:
F 类的 MRO : (<class '__main__.F'>, <class '__main__.E'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.D'>, <class '__main__.A'>, <class 'object'>)
执行 F 中的 funff()...
找到 funxx() 位于 C 中...
四、多继承重复调用
先看调用父类方法的普通使用:直接用类名.方法即可;但是在父类名字变化时还得回到子类去做相应修改,很麻烦
class A:
def __init__(self):
print("打印属性 a")
class B(A):
def __init__(self):
print("打印属性 b")
A.__init__(self) # super() 等同于 super(B, self)
class C(A):
def __init__(self):
print("打印属性 c")
A.__init__(self) # super() 等同于 super(C, self)
class D(B, C):
def __init__(self):
print("打印属性 d")
B.__init__(self)
C.__init__(self)
d = D()
#以下为输出结果
打印属性 d
打印属性 b
打印属性 a
打印属性 c
打印属性 a
可以看到A的构造函数重复调用了两次,造成资源浪费
再来看看使用super函数进行调用父类的方法
class A:
def __init__(self):
print("打印属性 a")
#
class B(A):
def __init__(self):
print("打印属性 b")
super().__init__() # super() 等同于 super(B, self)
class C(A):
def __init__(self):
print("打印属性 c")
super().__init__() # super() 等同于 super(C, self)
class D(B, C):
def __init__(self):
print("打印属性 d")
super(D, self).__init__()
d = D()
#输出结果
打印属性 d
打印属性 b
打印属性 c
打印属性 a
(1)强调一下:这里的super()没有参数,就默认都采用了本身类;比如A中的super()相当于super(A,self)
(2)在讲解一下整个程序的运行过程:初始化d对象,打印“打印属性 d”;然后super函数运行,调用了D类的__init__方法,但是这个self是d对象,所以这个顺序一直都是d对象的__mro__属性;打印“打印属性 b”;再调用super函数,按照d对象给出的顺序,下一个是C类的__init__方法,打印“打印属性 c”;继续调用super函数,最后才是A类的__init__方法,打印“打印属性 a”。
总结
以上就是关于super函数的使用讲解,谢谢大家;其中的代码来自(144条消息) Python super( ) 函数详解_小皇鱼的博客-CSDN博客_python super();但是讲解还是自己一个字一个字敲的。
文章出处登录后可见!