Python多重继承中的菱形继承


本文摘自php中文网,作者angryTom,侵删。

本篇文章介绍了Python多重继承中,菱形继承的的用法,希望对学习Python中的朋友有帮助!

继承是面向对象编程的一个重要的方式,通过继承,子类就可以扩展父类的功能。在python中一个类能继承自不止一个父类,这叫做python的多重继承(Multiple Inheritance )。

语法

1

2

class SubclassName(BaseClass1, BaseClass2, BaseClass3, ...):

    pass

菱形继承

推荐学习:Python视频教程

在多层继承和多继承同时使用的情况下,就会出现复杂的继承关系,多重多继承。

其中,就会出现菱形继承。如下图所示。mark

在这种结构中,在调用顺序上就出现了疑惑,调用顺序究竟是以下哪一种顺序呢

  • D->B->A->C(深度优先)
  • D->B->C->A(广度优先)

下面我们来解答下这个问题。

举个例子来看下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

class A():

    def __init__(self):

        print('init A...')

        print('end A...')

 

class B(A):

    def __init__(self):

        print('init B...')

        A.__init__(self)

        print('end B...')

 

class C(A):

    def __init__(self):

        print('init C...')

        A.__init__(self)

        print('end C...')

 

class D(B, C):

    def __init__(self):

        print('init D...')

        B.__init__(self)

        C.__init__(self)

        print('end D...')

 

if __name__ == '__main__':

    D()

输出结果

1

2

3

4

5

6

7

8

9

10

init D...

init B...

init A...

end A...

end B...

init C...

init A...

end A...

end C...

end D...

从输出结果中看,调用顺序为:D->B->A->C->A。可以看到,B、C共同继承于A,A被调用了两次。A没必要重复调用两次。

其实,上面问题的根源都跟MRO有关,MRO(Method Resolution Order)也叫方法解析顺序,主要用于在多重继承时判断调的属性来自于哪个类,其使用了一种叫做C3的算法,其基本思想时在避免同一类被调用多次的前提下,使用广度优先和从左到右的原则去寻找需要的属性和方法。

那么如何避免顶层父类中的某个方法被多次调用呢,此时就需要super()来发挥作用了,super本质上是一个类,内部记录着MRO信息,由于C3算法确保同一个类只会被搜寻一次,这样就避免了顶层父类中的方法被多次执行了,上面代码可以改为:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

class A():

    def __init__(self):

        print('init A...')

        print('end A...')

 

class B(A):

    def __init__(self):

        print('init B...')

        super(B, self).__init__()

        print('end B...')

 

class C(A):

    def __init__(self):

        print('init C...')

        super(C, self).__init__()

        print('end C...')

 

class D(B, C):

    def __init__(self):

        print('init D...')

        super(D, self).__init__()

        print('end D...')

 

if __name__ == '__main__':

    D()

输出结果:

1

2

3

4

5

6

7

8

init D...

init B...

init C...

init A...

end A...

end C...

end B...

end D...

可以看出,此时的调用顺序是D->B->C->A。即采用是广度优先的遍历方式。

补充内容

Python类分为两种,一种叫经典类,一种叫新式类。都支持多继承,但继承顺序不同。

  • 新式类:从object继承来的类。(如:class A(object)),采用广度优先搜索的方式继承(即先水平搜索,再向上搜索)。
  • 经典类:不从object继承来的类。(如:class A()),采用深度优先搜索的方式继承(即先深入继承树的左侧,再返回,再找右侧)。

Python2.x中类的是有经典类和新式类两种。Python3.x中都是新式类。

更多python相关教程,请关注python教程栏目。

以上就是Python多重继承中的菱形继承的详细内容,更多文章请关注木庄网络博客!!

相关阅读 >>

Python flask是什么

web自动化测试(三)selenium+beatuifulsoup

Python写温度转换

Python中不等于用什么来表示

Python怎么截图

自学Python可以做什么兼职

Python中类的创建与使用详解

Python中tuple指什么

Python是c语言开发的吗

使用Python如何对日志进行处理 (代码)

更多相关阅读请进入《Python》频道 >>




打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...