谈谈python super

python中的super是用来调用父类的方法吗?

答:不算错,但是不准确。某些情况下是这样的,某些情况下又不是。

用一句话说明super的作用是什么?

答:简单来说就是提供一个 MRO 以及一个 MRO 中的类 C , super() 将返回从 MRO 中 查找C类之后的第一个类的实例。

通过源代码更容易理解

def super(cls, inst):
    mro = inst.__class__.mro()
    return mro[mro.index(cls) + 1]

inst提供了一个MRO,cls提供了MRO中的类C,而super返回的就是从inst提供的MRO中查找,cls这个类后面一个类的实例。

注:不光可以通过inst提供MRO,也可以通过类提供MRO。那么super的形式就是super(A, B),A和B都是类,而不是实例。意义也是一样。

B提供了一个MRO,A提供了MRO中的类C,而super返回的就是从B提供的MRO中查找,A这个类后面一个类的实例。

举个实际例子:

这里环境是python3

class Root(object):
    def __init__(self):
        print("this is Root")

class B(Root):
    def __init__(self):
        print("enter B")
        # print(self)  # this will print <__main__.D object at 0x...>
        super(B, self).__init__()
        print("leave B")

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

class D(B, C):
    pass

d = D()
print(d.__class__.__mro__)

d的MRO:

(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.Root'>, <type 'object'>)
  1. D类实例化时,执行__init__()函数,因为 D 没有定义 init,所以会在d的 MRO 中找下一个类,去查看它有没有定义 init,也就是去调用 B 的 init
  2. 调用B的__init__,打印enter B,调用super(B, self).init(),根据我们刚刚讲的super的定义,其实他并不是查找父类的实例,而是查找self这个实例(这个实例是D的实例d,而不是B类的实例)的MRO中,B类之后的下一个类的实例,这里就是C类,所以会打印enter C,然后调用super(C, self).init()方法。
  3. 同理,调用super(C, self).init()方法,是根据self这个实例(这个实例是D的实例d,而不是B类的实例)的MRO中,C类之后的下一个类的实例,就是Root类的实例。所以打印this is Root。然后打印leave C, leave B。

备注:
super 是个类
当我们调用 super() 的时候,实际上是实例化了一个 super 类。你没看错, super 是个类,既不是关键字也不是函数等其他数据结构:

>>> class A:pass
... 
>>> s=super(A)
>>> type(s)
<class 'super'>
>>> isinstance(s,A)
False
>>> isinstance(s,super)
True
>>> 

来看一下super继承的列子

class A:
    def __init__(self):
        self.n = 2

    def add(self, m):
        # 第四步
        # 来自 D.add 中的 super
        # self == d, self.n == d.n == 5
        print('self is {0} @A.add'.format(self))
        self.n += m
        # d.n == 7


class B(A):
    def __init__(self):
        self.n = 3

    def add(self, m):
        # 第二步
        # 来自 D.add 中的 super
        # self == d, self.n == d.n == 5
        print('self is {0} @B.add'.format(self))
        # 等价于 suepr(B, self).add(m)
        # self 的 MRO 是 [D, B, C, A, object]
        # 从 B 之后的 [C, A, object] 中查找 add 方法
        super().add(m)

        # 第六步
        # d.n = 11
        self.n += 3
        # d.n = 14

class C(A):
    def __init__(self):
        self.n = 4

    def add(self, m):
        # 第三步
        # 来自 B.add 中的 super
        # self == d, self.n == d.n == 5
        print('self is {0} @C.add'.format(self))
        # 等价于 suepr(C, self).add(m)
        # self 的 MRO 是 [D, B, C, A, object]
        # 从 C 之后的 [A, object] 中查找 add 方法
        super().add(m)

        # 第五步
        # d.n = 7
        self.n += 4
        # d.n = 11


class D(B, C):
    def __init__(self):
        self.n = 5

    def add(self, m):
        # 第一步
        print('self is {0} @D.add'.format(self))
        # 等价于 super(D, self).add(m)
        # self 的 MRO 是 [D, B, C, A, object]
        # 从 D 之后的 [B, C, A, object] 中查找 add 方法
        super().add(m)

        # 第七步
        # d.n = 14
        self.n += 5
        # self.n = 19

d = D()
d.add(2)
print(d.n)

结果是:

self is <__main__.D object at 0x10ce10e48> @D.add
self is <__main__.D object at 0x10ce10e48> @B.add
self is <__main__.D object at 0x10ce10e48> @C.add
self is <__main__.D object at 0x10ce10e48> @A.add
19

我们看到并不是按照继承的顺序打印的,不是按照D继承B,C,B和C有继承A,D,B,A,C,A。

参考:
https://mozillazg.com/2016/12/python-super-is-not-as-simple-as-you-thought.html
https://laike9m.com/blog/li-jie-python-super,70/

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python 中,`super()` 是一个内置函数,用于调用父类方法。`super()` 函数可以在子类中调用父类中被重写的方法,从而实现对父类方法的继承并且进行扩展。使用 `super()` 函数的好处是,它能够动态地查找当前子类继承链中的下一个类,从而允许我们更加灵活地组织代码,同时也能够避免硬编码父类名称所带来的问题。 在使用 `super()` 函数时,需要注意以下几点: - `super()` 函数需要传入两个参数,第一个参数是当前子类的类名,第二个参数是当前对象。 - `super()` 函数返回的是一个代理对象,而不是真正的父类对象,因此需要通过该代理对象调用父类的方法。 - 如果在多重继承的情况下,需要在所有的父类中使用 `super()` 函数,否则可能会出现调用错误的情况。 下面是一个简单的示例代码,演示了如何使用 `super()` 函数调用父类方法: ```python class Animal: def __init__(self, name): self.name = name def eat(self): print("{} is eating...".format(self.name)) class Dog(Animal): def __init__(self, name, breed): super().__init__(name) self.breed = breed def eat(self): super().eat() print("{} is a {} dog".format(self.name, self.breed)) dog = Dog("Buddy", "Golden Retriever") dog.eat() ``` 在上面的代码中,我们定义了一个 `Animal` 类和一个 `Dog` 类,`Dog` 类继承自 `Animal` 类。`Animal` 类中有一个 `eat()` 方法,用于描述动物的进食方式。`Dog` 类中重写了 `eat()` 方法,并使用 `super()` 函数来调用父类 `Animal` 的 `eat()` 方法。当我们创建 `Dog` 类的对象 `dog` 并调用 `eat()` 方法时,会输出以下结果: ``` Buddy is eating... Buddy is a Golden Retriever dog ``` 可以看到,在 `Dog` 类的 `eat()` 方法中,首先调用了父类 `Animal` 的 `eat()` 方法,然后再输出狗的品种。这样做的好处是,当我们修改父类 `Animal` 的 `eat()` 方法时,子类 `Dog` 的方法也会随之改变,而不需要手动修改子类的方法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值