您的位置:首页 > 编程语言 > Python开发

python多继承(新式类)一

2016-03-14 23:15 316 查看
最近在学习python的多重继承。

先来了解下多重继承的概念,所谓多重继承,是指python的类可以有两个以上父类,也即有类A,类B,类C,C同时继承类A与类B,此时C中可以使用A与B中的属性与方法。那么问题来了,如果A与B中具有相同名字的方法,这个时候python怎么调用的会是哪个方法呢?

 

举个例子:

class A(object):
def __init__(self):
pass

def foo(self):

print 'A foo'

class B(object):
def __init__(self):
pass

def foo(self):

print 'B foo'

class C(A,B):

def __init__(self):

pass

testc = C()

testc.foo()

实际上打印出来的信息是 A foo,这就说明了调用的是A中的方法。其实在python2.2之后,多继承中基类的寻找顺序是一种广度优先算法,称之为C3的算法(后续博客我会简单介绍下C3算法)。而python2.2之前,使用的是深度优先算法来寻找基类方法。在类C的继承关系中,按照广度优先算法,则会先找到靠近C的基类A,在A中找到foo方法之后,就直接返回了,因此即使后面的基类B中也有foo方法,但是这里不会引用它。

更加清晰的多继承例子:

class A(object):
def foo(self):
print 'A foo'

class B(object):
def foo(self):
print 'B foo'
def bar(self):
print 'B bar'

class C1(A,B):
pass

class C2(A,B):
def bar(self):
print 'C2-bar'

class D(C1,C2):
pass

if __name__ =='__main__':
print D.__mro__ #只有新式类有__mro__属性,告诉查找顺序是怎样的
d=D()
d.foo()
d.bar()
执行的结果为:


(<class '__main__.D'>, <class '__main__.C1'>, <class '__main__.C2'>, <class '__main__.A'>, <class '__main__.B'>, <type 'object'>)

A foo (实际上搜索顺序为D=>C1=>A)

C2 bar(实际上搜索顺序为D=>C1=>C2)

可以看到,foo找到的是A类中的方法,bar找到的是C2中的方法。

其实新式类的搜索方法是采用了“广度优先”的方式去查找属性。

只有新式类有__mro__属性,该属性标记了python继承层次中父类查找的顺序,python多重继承机制中就是按照__mro__的顺序进行查找,一旦找到对应属性,则查找马上返回。

经过上面的__mro__输出可以发现,D类的继承查找路径为:D=>C1=>C2=>A=>B=>object,通过该查找路径,foo方法将会调用A的foo方法,、bar方法将调用C2的方法,通过实际实验调用,查看输出内容确实与__mro__顺序一样。

实际上打印出来的信息是 A foo,这就说明了调用的是A中的方法。其实在python2.2之后,多继承中基类的寻找顺序是一种广度优先算法,称之为C3的算法(后续博客我会简单介绍下C3算法)。而python2.2之前,使用的是深度优先算法来寻找基类方法。在类C的继承关系中,按照广度优先算法,则会先找到靠近C的基类A,在A中找到foo方法之后,就直接返回了,因此即使后面的基类B中也有foo方法,但是这里不会引用它。

其实新式类的搜索方法是采用了“广度优先”的方式去查找属性。

只有新式类有__mro__属性,该属性标记了python继承层次中父类查找的顺序,python多重继承机制中就是按照__mro__的顺序进行查找,一旦找到对应属性,则查找马上返回。

经过上面的__mro__输出可以发现,D类的继承查找路径为:D=>C1=>C2=>A=>B=>object,通过该查找路径,foo方法将会调用A的foo方法,、bar方法将调用C2的方法,通过实际实验调用,查看输出内容确实与__mro__顺序一样。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  python 继承 多继承