多继承以及MRO顺序

  1. 单独调用父类的方法

coding=utf-8

print(“*多继承使用类名.init 发生的状态*“)
class Parent(object):
def init(self, name):
print(‘parent的init开始被调用’)
self.name = name
print(‘parent的init结束被调用’)

class Son1(Parent):
def init(self, name, age):
print(‘Son1的init开始被调用’)
self.age = age
Parent.init(self, name)
print(‘Son1的init结束被调用’)

class Son2(Parent):
def init(self, name, gender):
print(‘Son2的init开始被调用’)
self.gender = gender
Parent.init(self, name)
print(‘Son2的init结束被调用’)

class Grandson(Son1, Son2):
def init(self, name, age, gender):
print(‘Grandson的init开始被调用’)
Son1.init(self, name, age) # 单独调用父类的初始化方法
Son2.init(self, name, gender)
print(‘Grandson的init结束被调用’)

gs = Grandson(‘grandson’, 12, ‘男’)
print(‘姓名:’, gs.name)
print(‘年龄:’, gs.age)
print(‘性别:’, gs.gender)

print(“*多继承使用类名.init 发生的状态*\n\n”)
运行结果:

*多继承使用类名.init 发生的状态*
Grandson的init开始被调用
Son1的init开始被调用
parent的init开始被调用
parent的init结束被调用
Son1的init结束被调用
Son2的init开始被调用
parent的init开始被调用
parent的init结束被调用
Son2的init结束被调用
Grandson的init结束被调用
姓名: grandson
年龄: 12
性别: 男
*多继承使用类名.init 发生的状态*
2. 多继承中super调用有所父类的被重写的方法
print(“*多继承使用super().init 发生的状态*“)
class Parent(object):
def init(self, name, *args, **kwargs): # 为避免多继承报错,使用不定长参数,接受参数
print(‘parent的init开始被调用’)
self.name = name
print(‘parent的init结束被调用’)

class Son1(Parent):
def init(self, name, age, *args, **kwargs): # 为避免多继承报错,使用不定长参数,接受参数
print(‘Son1的init开始被调用’)
self.age = age
super().init(name, *args, **kwargs) # 为避免多继承报错,使用不定长参数,接受参数
print(‘Son1的init结束被调用’)

class Son2(Parent):
def init(self, name, gender, *args, **kwargs): # 为避免多继承报错,使用不定长参数,接受参数
print(‘Son2的init开始被调用’)
self.gender = gender
super().init(name, *args, **kwargs) # 为避免多继承报错,使用不定长参数,接受参数
print(‘Son2的init结束被调用’)

class Grandson(Son1, Son2):
def init(self, name, age, gender):
print(‘Grandson的init开始被调用’)
# 多继承时,相对于使用类名.init方法,要把每个父类全部写一遍
# 而super只用一句话,执行了全部父类的方法,这也是为何多继承需要全部传参的一个原因
# super(Grandson, self).init(name, age, gender)
super().init(name, age, gender)
print(‘Grandson的init结束被调用’)

print(Grandson.mro)

gs = Grandson(‘grandson’, 12, ‘男’)
print(‘姓名:’, gs.name)
print(‘年龄:’, gs.age)
print(‘性别:’, gs.gender)
print(“*多继承使用super().init 发生的状态*\n\n”)
运行结果:

*多继承使用super().init 发生的状态*
(

print(‘性别:’, gs.gender)

print(“*单继承使用super().init 发生的状态*\n\n”)
总结
super().init相对于类名.init,在单继承上用法基本无差
但在多继承上有区别,super方法能保证每个父类的方法只会执行一次,而使用类名的方法会导致方法被执行多次,具体看前面的输出结果
多继承时,使用super方法,对父类的传参数,应该是由于python中super的算法导致的原因,必须把参数全部传递,否则会报错
单继承时,使用super方法,则不能全部传递,只能传父类方法所需的参数,否则会报错
多继承时,相对于使用类名.init方法,要把每个父类全部写一遍, 而使用super方法,只需写一句话便执行了全部父类的方法,这也是为何多继承需要全部传参的一个原因
小试牛刀(以下为面试题)
以下的代码的输出将是什么? 说出你的答案并解释。

class Parent(object):
x = 1

class Child1(Parent):
pass

class Child2(Parent):
pass

print(Parent.x, Child1.x, Child2.x)
Child1.x = 2
print(Parent.x, Child1.x, Child2.x)
Parent.x = 3
print(Parent.x, Child1.x, Child2.x)
答案, 以上代码的输出是:

1 1 1
1 2 1
3 2 3
使你困惑或是惊奇的是关于最后一行的输出是 3 2 3 而不是 3 2 1。为什么改变了 Parent.x 的值还会改变 Child2.x 的值,但是同时 Child1.x 值却没有改变?

这个答案的关键是,在 Python 中,类变量在内部是作为字典处理的。如果一个变量的名字没有在当前类的字典中发现,将搜索祖先类(比如父类)直到被引用的变量名被找到(如果这个被引用的变量名既没有在自己所在的类又没有在祖先类中找到,会引发一个 AttributeError 异常 )。

因此,在父类中设置 x = 1 会使得类变量 x 在引用该类和其任何子类中的值为 1。这就是因为第一个 print 语句的输出是 1 1 1。

随后,如果任何它的子类重写了该值(例如,我们执行语句 Child1.x = 2),然后,该值仅仅在子类中被改变。这就是为什么第二个 print 语句的输出是 1 2 1。

最后,如果该值在父类中被改变(例如,我们执行语句 Parent.x = 3),这个改变会影响到任何未重写该值的子类当中的值(在这个示例中被影响的子类是 Child2)。这就是为什么第三个 print 输出是 3 2 3。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值