python的super函数详解

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();但是讲解还是自己一个字一个字敲的。

文章出处登录后可见!

已经登录?立即刷新

共计人评分,平均

到目前为止还没有投票!成为第一位评论此文章。

(0)
社会演员多的头像社会演员多普通用户
上一篇 2023年8月23日
下一篇 2023年8月23日

相关推荐

此站出售,如需请站内私信或者邮箱!