python中super方法的作用_python类中super()和__init__()的区别

最近有同学问我关于Python类中的super()和__init__()共同点和不同点的问题, 我今天把它们两个的异同点总结了一下,希望可以帮助遇到同样困惑的同学。

单继承时super()和__init__()实现的功能是类似的class Base(object):

def __init__(self):

print 'Base create'

class childA(Base):

def __init__(self):

print 'creat A ',

Base.__init__(self)

class childB(Base):

def __init__(self):

print 'creat B ',

super(childB, self).__init__()

base = Base()

a = childA()

b = childB()

输出结果:Base create

creat A Base create

creat B Base create

区别是使用super()继承时不用显式引用基类。

super()只能用于新式类中

把基类改为旧式类,即不继承任何基类class Base():

def __init__(self):

print 'Base create'

执行时,在初始化b时就会报错:super(childB, self).__init__()

TypeError: must be type, not classobj

super不是父类,而是继承顺序的下一个类

在多重继承时会涉及继承顺序,super()相当于返回继承顺序的下一个类,而不是父类,类似于这样的功能:def super(class_name, self):

mro = self.__class__.mro()

return mro[mro.index(class_name) + 1]

mro()用来获得类的继承顺序。 例如:class Base(object):

def __init__(self):

print 'Base create'

class childA(Base):

def __init__(self):

print 'enter A '

# Base.__init__(self)

super(childA, self).__init__()

print 'leave A'

class childB(Base):

def __init__(self):

print 'enter B '

# Base.__init__(self)

super(childB, self).__init__()

print 'leave B'

class childC(childA, childB):

pass

c = childC()

print c.__class__.__mro__

输出结果如下:enter A

enter B

Base create

leave B

leave A

(, , , , )

supder和父类没有关联,因此执行顺序是A —> B—>—>Base

执行过程相当于:初始化childC()时,先会去调用childA的构造方法中的 super(childA, self).__init__(), super(childA, self)返回当前类的继承顺序中childA后的一个类childB;然后再执行childB().__init__(),这样顺序执行下去。

在多重继承里,如果把childA()中的 super(childA, self).__init__() 换成Base._init_(self),在执行时,继承childA后就会直接跳到Base类里,而略过了childB:enter A

Base create

leave A

(, , , , )

从super()方法可以看出,super()的第一个参数可以是继承链中任意一个类的名字,

如果是本身就会依次继承下一个类;

如果是继承链里之前的类便会无限递归下去;

如果是继承链里之后的类便会忽略继承链汇总本身和传入类之间的类;

比如将childA()中的super改为:super(childC, self).__init__(),程序就会无限递归下去。 如:File "test.py", line 12, in __init__

super(childC, self).__init__()

File "test.py", line 12, in __init__

super(childC, self).__init__()

File "test.py", line 12, in __init__

super(childC, self).__init__()

File "test.py", line 12, in __init__

super(childC, self).__init__()

File "test.py", line 12, in __init__

super(childC, self).__init__()

File "test.py", line 12, in __init__

super(childC, self).__init__()

File "test.py", line 12, in __init__

super(childC, self).__init__()

File "test.py", line 12, in __init__

super(childC, self).__init__()

File "test.py", line 12, in __init__

super(childC, self).__init__()

RuntimeError: maximum recursion depth exceeded while calling a Python object

super()可以避免重复调用

如果childA基础Base, childB继承childA和Base,如果childB需要调用Base的__init__()方法时,就会导致__init__()被执行两次:class Base(object):

def __init__(self):

print 'Base create'

class childA(Base):

def __init__(self):

print 'enter A '

Base.__init__(self)

print 'leave A'

class childB(childA, Base):

def __init__(self):

childA.__init__(self)

Base.__init__(self)

b = childB()

Base的__init__()方法被执行了两次enter A

Base create

leave A

Base create

使用super()是可避免重复调用class Base(object):

def __init__(self):

print 'Base create'

class childA(Base):

def __init__(self):

print 'enter A '

super(childA, self).__init__()

print 'leave A'

class childB(childA, Base):

def __init__(self):

super(childB, self).__init__()

b = childB()

print b.__class__.mro()

enter A

Base create

leave A

[, , , ]

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值