python类继承(super多类继承)

1. python2和python3中定义类的形式不同

python3中只有只有新式类

class A(object): # python3中定义新式类
class A: # python3中定义新式类(省略object)

python2中有经典类和新式类区别:

class A(object): # python2中定义新式类
class A: # python2中定义经典类

新式类和经典类只要区别在继承搜索顺序上,
-- 经典类,搜索深度优先,先深入继承树左侧查找,然后返回,开始右侧查找
-- 新式类,搜索广度优先,先水平查找,再向上查找
python2-新式类:super(类, self)必须显示的申明,python3-默认自己时可省略而使用super()


python2-新式类例子

class A(object):
    def __init__(self):
        print("A")


class B(A):
    def __init__(self):
        print("B")
        super(B,self).__init__()


class C(A):
    def __init__(self):
        print("C")
        super(C,self).__init__()


class D(B, C):
    def __init__(self):
        print("D")
        super(D,self).__init__()

d = D()
print(D.mro())

新式类——继承树查找——广度模式,D —> B —> C —> A

>>> d = D()
D
B
C
A
>>> print(D.mro())
[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <type 'object'>]

python2经典类

class A:
    def __init__(self):
        print("A")


class B(A):
    def __init__(self):
        print("B")
        super(B,self).__init__()


class C(A):
    def __init__(self):
        print("C")
        super(C,self).__init__()


class D(B, C):
    def __init__(self):
        print("D")
        super(D,self).__init__()

结果:经典类没有mro属性,也就不能用super(class, self)来调用

>>> d = D()
D
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<stdin>", line 4, in __init__
TypeError: super() argument 1 must be type, not classobj
>>> D.mro()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: class D has no attribute 'mro'

2. 类继承的两种方式

** 显示地调用方法:类.__init__(self)
** 通过 super() 调用

2.1 单继承下,两者结果一样
class A(object):
    def __init__(self):
        print("A")


class B(A):
    def __init__(self):
        print("B")
        A.__init__(self)

b = B()


class B(A):
    def __init__(self):
        print("B")
        super().__init__()

b = B()

两者方法的结果一样

B
A
B
A
2.2 多继承下

显示调用方法在多继承模式下存在的问题父类被调用不止一次

class A(object):
    def __init__(self):
        print("A")


class B(A):
    def __init__(self):
        print("B")
        A.__init__(self)


class C(A):
    def __init__(self):
        print("C")
        A.__init__(self)


class D(B, C):
    def __init__(self):
        print("D")
        B.__init__(self)
        C.__init__(self)

d = D()

结果 A 类被调用了两次

D
B
A
C
A

使用 super() 方法,相同的类只被调用一次,python2 中使用 super(class, self).__init__(),而python3 中默认可不写,直接使用super().__init__()

class A(object):
    def __init__(self):
        print("A")


class B(A):
    def __init__(self):
        print("B")
        super().__init__()


class C(A):
    def __init__(self):
        print("C")
        super().__init__()


class D(B, C):
    def __init__(self):
        print("D")
        super().__init__()

d = D()

结果如下:

D
B
C
A

这里需要注意的一点,如果部分类使用super()方法,会出现继承不全等问题,如下:
其中B、C类显示继承,导致结果出问题

class A(object):
    def __init__(self):
        print("A")


class B(A):
    def __init__(self):
        print("B")
        A.__init__(self)


class C(A):
    def __init__(self):
        print("C")
        A.__init__(self)


class D(B, C):
    def __init__(self):
        print("D")
        super().__init__()

d = D()

结果:

D
B
A

tips: ① 不要混合使用类.__init__(self)super(),② 尽量使用 super()方法

3. super调用顺序 —— mro

3.1 查看类的继承顺序

mro(method resolution order):表示类继承体系中的成员顺序。查看方法:类.mro()
回到前面那个例子:

class A(object):
    def __init__(self):
        print("A")


class B(A):
    def __init__(self):
        print("B")
        A.__init__(self)


class C(A):
    def __init__(self):
        print("C")
        A.__init__(self)


class D(B, C):
    def __init__(self):
        print("D")
        super().__init__()

print(D.mro())

结果表示对于D类的继承顺序为: D —> B —> C —> A,这与上述的结果相符。

[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
3.2 super(cls, self)类指定,默认为当前类
class A(object):
    def __init__(self):
        print("A")


class B(A):
    def __init__(self):
        print("B")
        super().__init__()


class C(A):
    def __init__(self):
        print("C")
        super().__init__()

使用默认类此时 super().__init__() == super(D, self).__init__()

class D(B, C):
    def __init__(self):
        print("D")
        super(D,self).__init__()

d = D()
print(D.mro())

结果如下:

D
B
C
A
[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]

修改默认为类为B,

class D(B, C):
    def __init__(self):
        print("D")
        super(B, self).__init__()

d = D()
print(D.mro())

结果如下:发现B不见了。 为什么?

  • 这里super(B, self)表示:
    -- self这里指D自身,也即将D的继承顺序 [D, B, C, A, object] 传递给super()
    -- B表示从B的下一顺序开始,这里从 C —> A —> object
    -- 将第一个找到的__init__函数绑定到self对象,并返回
D
C
A
[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容

  • Python进阶-继承中的MRO与super @(Python)[python, python进阶] [TOC] ...
    理查德成阅读 1,551评论 2 10
  • property、魔法属性和魔法方法、多重继承和多继承 1.5 property 学习目标 1. 能够说出什么要...
    Cestine阅读 791评论 0 1
  • 说到 super, 大家可能觉得很简单呀,不就是用来调用父类方法的嘛。如果真的这么简单的话也就不会有这篇文章了,且...
    流光汐舞阅读 840评论 0 0
  • Python支持多继承,与C++一样都会出现一种问题:子类继承的多个父类又继承了同一个父类,这时就有可能会出现父类...
    杰克慢阅读 396评论 0 3
  • 1.1面向对象 面向对象(object-oriented ;简称: OO)至今还没有统一的概念 我这里把它定义为:...
    TENG书阅读 564评论 0 0