编写类之四再论继承

在上一讲代码的基础上,做进一步修改,成为了如下程序,请看官研习这个程序:

  1. #!/usr/bin/env python
  2. #coding:utf-8
  3. class Person:
  4. def __init__(self, name, email):
  5. self.name = name
  6. self.email = email
  7. class Programmer(Person):
  8. def __init__(self, name,email,lang, system, website):
  9. Person.__init__(self,name,email)
  10. self.lang = lang
  11. self.system = system
  12. self.website = website
  13. class Pythoner(Programmer):
  14. def __init__(self,name,email):
  15. Programmer.__init__(self,name,email,"python","Ubuntu","qiwsir.github.io")
  16. if __name__=="__main__":
  17. writer = Pythoner("qiwsir","qiwsir@gmail.com")
  18. print "name=",writer.name
  19. print "lang=",writer.lang
  20. print "email=",writer.email
  21. print "system=",writer.system
  22. print "website=",writer.website
  23. #运行结果
  24. name= qiwsir
  25. lang= python
  26. email= qiwsir@gmail.com
  27. system= Ubuntu
  28. website= qiwsir.github.io

对结果很满意,再看程序中的继承关系:Pythoner <— Programmer <— Person,从上面的过程中不难看出,继承能够减少代码重复,是的代码更简练。另外,在继承的时候,也可以在函数中对参数进行默认赋值。

为了能够突出继承问题的探究,还是用那种简单的类来做实验。

多余的B

  1. #!/usr/bin/env python
  2. #coding:utf-8
  3. class A:
  4. def __init__(self):
  5. print "aaa"
  6. class B(A):
  7. pass
  8. if __name__=="__main__":
  9. a = A()
  10. b = B()
  11. #运行结果
  12. aaa
  13. aaa

B继承A,没有任何修改地继承,B就可以不用写任何东西了,或者说B本质上就是一个多余。在真实的编程过程中,没有这样写的,这里仅仅是为了向看官展示一下继承的含义罢了。

  1. ##首个继承有效
  2. #!/usr/bin/env python
  3. #coding:utf-8
  4. class A:
  5. def __init__(self):
  6. print "aaa"
  7. class B:
  8. def __init__(self):
  9. print "bbb"
  10. class C1(A,B):
  11. pass
  12. class C2(B,A):
  13. pass
  14. if __name__=="__main__":
  15. print "A--->",
  16. a = A()
  17. print "B--->",
  18. b = B()
  19. print "C1(A,B)--->",
  20. c1 = C1()
  21. print "C2(B,A)--->",
  22. c2 = C2()
  23. #运行结果
  24. A---> aaa
  25. B---> bbb
  26. C1(A,B)---> aaa
  27. C2(B,A)---> bbb

列位看官是否注意了,类C1继承了两个类A,B;类C2也继承了两个类,只不过书写顺序有点区别(B,A)。从运行结果可以看出,当子类继承多个父类的时候,对于构造函数__init__(),只有第一个能够被继承,第二个就等掉了。所以,一般情况下,不会在程序中做关于构造函数的同时多个继承,不过可以接力继承,就如同前面那个比较真实的代码一样。

其它方法的继承

  1. #!/usr/bin/env python
  2. #coding:utf-8
  3. class A:
  4. def __init__(self):
  5. print "aaa"
  6. def amethod(self):
  7. print "method a"
  8. class B(A):
  9. def __init__(self):
  10. print "bbb"
  11. if __name__=="__main__":
  12. print "A--->"
  13. a = A()
  14. a.amethod()
  15. print "B--->"
  16. b = B()
  17. b.amethod()
  18. #运行结果
  19. A--->
  20. aaa
  21. method a
  22. B--->
  23. bbb
  24. method a

为了说明白上面的情况,还是画了一张图,不过,我画完之后,就后悔了,看这张图好像更糊涂了。怎么着也画了,还是贴出来,如果能够协助理解更好了。

编写类之四再论继承 - 图1

A的实例和调用,就不多说了。重点看B,类B继承了A,同时,B在构造函数中自己做了规定,也就是B的构造函数是按照B的意愿执行,不执行A的内容,但是,A还有一个amethod(self)方法,B则继承了这个方法。当通过类B的实例调用这个方法的时候,就能够成功了:b.amethod()

这就是方法的继承和调用方法。

所谓继承,就是从下到上一级一级地找相应的继承对象,找到了就继承之。如果有同名的怎么办?按照什么顺序找呢?

应用网上的一段:

在Python中,可以進行多重繼承,這個時候要注意搜尋的順序,是從子類別開始,接著是同一階層父類別由左至右搜尋,再至更上層同一階層父類別由左至右搜尋,直到達到頂層為止。

代码举例:

  1. class A(object):
  2. def method1(self):
  3. print('A.method1')
  4. def method2(self):
  5. print('A.method2')
  6. class B(A):
  7. def method3(self):
  8. print('B.method3')
  9. class C(A):
  10. def method2(self):
  11. print('C.method2')
  12. def method3(self):
  13. print('C.method3')
  14. class D(B, C):
  15. def method4(self):
  16. print('C.method4')
  17. d = D()
  18. d.method4() # 在 D 找到,C.method4
  19. d.method3() # 以 D->B 順序找到,B.method3
  20. d.method2() # 以 D->B->C 順序找到,C.method2
  21. d.method1() # 以 D->B->C->A 順序找到,A.method1

务必请真正的学习者要对照每个类的每个方法,依次找到相应的输出结果。从而理解继承的顺序。学习,就要点滴积累。