python子类如何调用父类属性_python类与对象详解(2):子类调用父类方法的两种方式...

当子类继承父类的一些方法并且子类已经覆盖此方法时,我们如何调用父类中的函数?这里提供两种调用方式:

通过在子类中实例化父类的方式调用父类函数

class Base:

def __init__(self):

print("Base.__init__")

class A(Base):

Base.__init__(self)

print("A.__init__")

这种方法比较奔放,对于大多数还是正常的,但是对于多重继承的时候会重复调用父类。解决办法就是使用第二种方法进行调用。

通过supper调用父类函数

class Base:

def __init__(self):

print("Base.__init__")

class A(Base):

def __init__(self):

Base.__init__(self)

print("A.__init__")

class B(Base):

def __init__(self):

Base.__init__(self)

print("B.__init__")

class C(A, B):

def __init__(self):

A.__init__(self)

B.__init__(self)

print("C.__init__")

运行一下代码:

>>>c = C()

Base.__init__

A.__init__

Base.__init__

B.__init__

C.__init__

这里可以清晰的看到当我们进行多重继承的时候父类被多次调用,看似没有任何问题,当时当代码复杂到一定程度时会摸不着头脑,产生的错误莫名其妙。我们再使用supper进行相同的操作:

class Base:

def __init__(self):

print("Base.__init__")

class A(Base):

def __init__(self):

supper().__init__()

print("A.__init__")

class B(Base):

def __init__(self):

supper().__init__()

print("B.__init__")

class C(A, B):

def __init__(self):

supper().__init__()

print("C.__init__")

运行代码:

>>> c = C()

Base.__init__

A.__init__

B.__init__

C.__init__

从得到的结果跟上面的代码运行结果进行对比可以得出以下结果:

当我们使用supper进行父类方法调用时,只会调用一次父类的方法

我们再来看看这个调用是怎样实现的:

python进行继承时,针对每个定义的类,python都会计算出一个方法解析顺序列表(MRO)MRO只是简单地针对所有的基类进行现行排列。实现继承时,python会从MRO列表中最左边的类开始,从左到右进行查找,直到找到待查的属性时为止。这是python继承的机制,那么MRO又是怎样实现的?

简单来说这是针对父类的一种归并排序,需要满足3个约束:

先检查子类再检查父类

有多个父类时,按照MRO列表的顺序依次检查

如果下一个带选的类中出现两个或多个合法的选择,那么就从第一个父类中选择。

再来回顾刚才的例子:

MRO会控制最终遍历整个MRO列表,并且每个方法只会被调用一次。这里就是刚才我们使用supper时为何只调用一次父类方法的原因。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值