经典类与新式类、钻石继承(菱形继承)、多继承之“广度优先”“深度优先”

一、经典类与新式类(了解)

  新式类:

    1)凡是继承object的类或子孙类都是新式类

    2)在python3中所有的类都是新式类(默认继承object)

 

  经典类:

    1)必须在python2中才会有经典类与新式类之分

    2)在python2中,凡是没有继承object的类,都是经典类

 

# 新式类:
class User(object):
    pass

class Sub(User):
    pass


# 在python3中所有的类都是新式类(默认继承object)
class Bar:
    pass

print(Bar.__bases__)    # (<class 'object'>,)

 

 

二、新式类查找属性和方法的顺序(了解)

  在python3中提供了一个新式类查找属性和方法的顺序的内置方法:

    mro():会把当前类的继承关系列出来

    调用mro返回的是一个继承序列:

      super()的继承序列严格遵循mro继承序列

 

  新式类查找属性和方法的顺序:

    1)先从子类中查找

    2)若子类中没有,从父类中查找,如果继承多个父类的情况下:从左到右

    3)若父类中也没有,从object中查找

    4)若object中也没有则报错

class Father1:
    x = 10

class Father2:
    x = 20

class Father3:
    x = 30

# 多继承的情况下:从左到右
class Sub(Father1, Father2, Father3):
    # 注意:__int__  不是  __init__
    def __init__(self):
        print(super().x)


obj = Sub()
print(Sub.mro())
# 子类和父类中都没有这个方法,但是object中有,所以从object中找到.__dir__
print(Sub.__dir__)

  执行结果:

10
[<class '__main__.Sub'>, <class '__main__.Father1'>, <class '__main__.Father2'>, <class '__main__.Father3'>, <class 'object'>]
<method '__dir__' of 'object' objects>

 

 

三、钻石继承(菱形继承)

  继承多个父类的情况下会造成"钻石继承"

 

  例:多继承

class A(object):
    def test(self):
        print("from A")

class B(A):
    def test(self):
        print("from B")

class C(A):
    def test(self):
        print("from C")

class D(B):
    def test(self):
        print("from D")

class E(C):
    def test(self):
        print("from E")

class F(D, E):
    def test(self):
        print("from F")


print(F.mro())

  执行结果:

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

  继承结果图:

   因为整体形状像钻石或者像菱形,所以叫做钻石继承,也叫做菱形继承。(类似的形状也可以,前提是多继承)

 

 

四、类继承之广度优先、深度优先

  python 支持多继承,但对与经典类和新式类来说,多继承查找的顺序是不一样的。

  内置方法mro()的查找顺序:

    - 新式类

      从左到右,广度优先

 

    - 经典类

      从左到右,深度优先

 

  两种概念图:

 

  新式类广度优先演示:

class A(object):
    def test(self):
        print("from A")

class B(A):
    # def test(self):
        # print("from B")
    pass

class C(A):
    def test(self):
        print("from C")

class D(B):
    # def test(self):
    #     print("from D")
    pass

class E(C):
    def test(self):
        print("from E")

class F(D, E):
    # def test(self):
    #     print("from F")
    pass


obj = F()
print(obj.test())

  执行结果:

from E

 

  经典式深度优先不做展示。

posted @ 2020-10-24 23:31  chchcharlie、  阅读(384)  评论(0编辑  收藏  举报