一、概述

  Python中支持多继承,也就是一个子类可以继承多个父类/基类。当一个调用一个自身没有定义的属性时,它是按照何种顺序去父类中寻找的呢?尤其是当众多父类中都包含有同名的属性,这就涉及到新式类 和 经典类的区别。

二、多继承

  1. class Food(object):
  2.  
  3. def __init__(self, name, color):
  4. self.name = name
  5. self.color = color
  6.  
  7. def eatable(self):
  8. print("%s can be eaten." % self.name)
  9.  
  10. def appearance(self):
  11. print('The color of the %s is %s.' % (self.name, self.color))
  12.  
  13. class Fruits(object):
  14.  
  15. def __init__(self, name, nutrition):
  16. self.name = name
  17. self.nutrition = nutrition
  18.  
  19. def info(self):
  20. print("%s can supply much %s." % (self.name, self.nutrition))
  21.  
  22. class Salad(Fruits, Food): # 继承多个父类
  23.  
  24. def __init__(self, name, nutrition, color, tasty):
  25. super(Salad, self).__init__(name, nutrition)
  26. Food.__init__(self, name, color)
  27. self.tasty = tasty
  28.  
  29. def taste(self):
  30. print("%s is a little %s." % (self.name, self.tasty))
  31.  
  32. obj = Salad('orange', 'VC', 'orange', 'sour')
  33.  
  34. obj.eatable()
  35. obj.appearance()
  36. obj.info()
  37. obj.taste()

  上例中的Salad(Fruits,Food)继承了Fruits和Food两个父类。

supper()函数为新式类的方法,采用新式类要求最顶层的父类一定要继承于object,这样就可以用super()函数来调用父类的init()等函数。每个父类都执行且执行一次,并不会出现重复调用的情况。采用super()方法时,会自动找到第一个多继承中的第一父类。

  但是如果想要继续调用其它父类init()函数或两个父类的同名函数时,就要用经典类的调用方法了,即 父类名.__init__(self,参数),如上例。

三、经典类 VS 新式类的继承顺序

3.1 新式类

1、新式类定义时必须继承object类,继承了object类的就叫做 新式类

  1. class Fruits(object):
  2. '新式类'
  3. pass

  

2、采用super()函数类调用父类的 init()等函数

  1. super(子类名,self).__init__(参数1,参数2,..)

  

3、调用父类中相同属性或者方法的顺序

新式类的调用顺序为: 广度优先查询

子类先在自己的所有父类中从左至右查询,如果没有需要的方法或属性,再到本身父类的父类中去查询。

代码如下:

1.调用本身属性

  1. class A(object):
  2. def __init__(self):
  3. self.n = "A"
  4.  
  5. class B(A):
  6.  
  7. def __init__(self):
  8. super(B, self).__init__()
  9. self.n = "B"
  10.  
  11. class C(A):
  12.  
  13. def __init__(self):
  14. super(C, self).__init__()
  15. self.n = "C"
  16.  
  17. class D(B, C):
  18.  
  19. def __init__(self):
  20. super(D, self).__init__()
  21. self.n = "D"
  22.  
  23. d = D()
  24. print(d.n)
  25.  
  26. #输出
  27. D

2.注释D中的代码,获得B

  1. class A(object):
  2. def __init__(self):
  3. self.n = "A"
  4.  
  5. class B(A):
  6.  
  7. def __init__(self):
  8. super(B, self).__init__()
  9. self.n = "B"
  10.  
  11. class C(A):
  12.  
  13. def __init__(self):
  14. super(C, self).__init__()
  15. self.n = "C"
  16.  
  17. class D(B, C):
  18.  
  19. # def __init__(self):
  20. # super(D, self).__init__()
  21. # self.n = "D"
  22. pass
  23.  
  24. d = D()
  25. print(d.n)
  26.  
  27. #输出
  28. B

3.注释B中的代码,获得C

  1. class A(object):
  2. def __init__(self):
  3. self.n = "A"
  4.  
  5. class B(A):
  6.  
  7. # def __init__(self):
  8. # super(B, self).__init__()
  9. # self.n = "B"
  10. pass
  11.  
  12. class C(A):
  13.  
  14. def __init__(self):
  15. super(C, self).__init__()
  16. self.n = "C"
  17.  
  18. class D(B, C):
  19.  
  20. # def __init__(self):
  21. # super(D, self).__init__()
  22. # self.n = "D"
  23. pass
  24.  
  25. d = D()
  26. print(d.n)
  27.  
  28. #输出
  29. C

4.注释C中的代码,获得A

  1. class A(object):
  2. def __init__(self):
  3. self.n = "A"
  4.  
  5. class B(A):
  6.  
  7. # def __init__(self):
  8. # super(B, self).__init__()
  9. # self.n = "B"
  10. pass
  11.  
  12. class C(A):
  13. #
  14. # def __init__(self):
  15. # super(C, self).__init__()
  16. # self.n = "C"
  17. pass
  18.  
  19. class D(B, C):
  20.  
  21. # def __init__(self):
  22. # super(D, self).__init__()
  23. # self.n = "D"
  24. pass
  25.  
  26. d = D()
  27. print(d.n)
  28.  
  29. #输出
  30. A

3.2 经典类

1、经典类定义,什么都不继承

  1. class Fruit:
  2. '经典类'
  3. pass

2、继承父类的init()等函数或属性

  1. 父类名.__init__(self, 参数1,参数2,....)

  

3、调用父类中相同属性或者方法的顺序

在 Python3 中,多继承的查询顺序都是 广度优先查询

经典类的调用顺序为: 深度优先查询

子类会沿着父类的父类这样的顺序查询,如果都没有,会返回查找另一个父类。

代码如下:

1.调用本身的属性

  1. class A:#经典类
  2. def __init__(self):
  3. self.n = "A"
  4.  
  5. class B(A):
  6. pass
  7. def __init__(self):
  8. self.n = "B"
  9.  
  10. class C(A):
  11. def __init__(self):
  12. self.n = "C"
  13.  
  14. class D(B,C):
  15. def __init__(self):
  16. self.n = "D"
  17.  
  18. d = D()
  19. print(d.n)
  20.  
  21. #输出
  22. D
  23.  
  24. 全部代码

2. 注释D中的代码,获得B

  1. class A:
  2. def __init__(self):
  3. self.n = "A"
  4.  
  5. class B(A):
  6. def __init__(self):
  7. self.n = "B"
  8.  
  9. class C(A):
  10. def __init__(self):
  11. self.n = "C"
  12.  
  13. class D(B,C):
  14. pass
  15.  
  16. d = D()
  17. print(d.n)
  18.  
  19. #输出
  20. B

3.注释B中的代码,获得A

  1. class A:
  2. def __init__(self):
  3. self.n = "A"
  4.  
  5. class B(A):
  6. pass
  7.  
  8. class C(A):
  9. def __init__(self):
  10. self.n = "C"
  11.  
  12. class D(B,C):
  13. pass
  14.  
  15. d = D()
  16. print(d.n)
  17.  
  18. #输出
  19. A

4.注释A中的代码,获得C

  1. class A:
  2. pass
  3.  
  4. class B(A):
  5. pass
  6.  
  7. class C(A):
  8. def __init__(self):
  9. self.n = "C"
  10.  
  11. class D(B,C):
  12. pass
  13.  
  14. d = D()
  15. print(d.n)
  16.  
  17. #输出
  18. C

四、总结

1、新式类继承object类,经典类不继承任何类

2、新式类用super关键字继承构造方法,经典类用 父类.__init(self)来继承

3、新式类:广度优先查询,经典类:深度优先查询(因为新式类讲究的是新,所以要找最近的,最新的;然后经典的讲究古老,所以更远更深的)

4、值得注意的是,我们上面是在python2中做的,在python3中不管是经典类还是新式类,都是采用的是广度优先查询,已经废弃2中的深度查询了

新式类 VS 经典类的更多相关文章

  1. python中新式类和经典类的区别

    1).python在类中的定义在py2-3版本上是使用的有新式类和经典类两种情况,在新式类和经典类的定义中最主要的区别是在定义类的时候是否出现引用object;如:经典类:Class 类名::而新式类 ...

  2. Python新式类与经典类的区别

    1.新式类与经典类 在Python 2及以前的版本中,由任意内置类型派生出的类(只要一个内置类型位于类树的某个位置),都属于“新式类”,都会获得所有“新式类”的特性:反之,即不由任意内置类型派生出的类 ...

  3. python之继承、抽象类、新式类和经典类

    一.上节补充1.静态属性静态属性 : 类的属性,所有的对象共享这个变量 如果用对象名去修改类的静态属性:在对象的空间中又创建了一个属性,而不能修改类中属性的值 操作静态属性应该用类名来操作 例1:请你 ...

  4. python中的__new__与__init__,新式类和经典类(2.x)

    在python2.x中,从object继承得来的类称为新式类(如class A(object))不从object继承得来的类称为经典类(如class A()) 新式类跟经典类的差别主要是以下几点: 1 ...

  5. Python 新式类与经典类

    新式类,经典类 查询匹配 广度查询 横着对每个类进行查询 深度查询 无视平级类,直接寻找下级类 #python 3.0 #新式类 广度查询 #经典类 广度查询 #python 2.0 #新式类 广度查 ...

  6. 面向对象【day07】:新式类和经典类(八)

    本节内容 1.概述 2.类的多继承 3.经典类VS新式类 4.总结 一.概述 在python还支持多继承,但是一般我们很少用,有些语言干脆就不支持多继承,有多继承,就会带来两个概念,经典类和新式类,下 ...

  7. python-反射、新式类与经典类搜索的优先级

    preface include: getattr setattr delattr hasattr class webserver(object): def __init__(self,num): se ...

  8. python单例模式控制成只初始化一次,常规型的python单例模式在新式类和经典类中的区别。

    单例模式的写法非常多,但常规型的单例模式就是这样写的,各种代码可能略有差异,但核心就是要搞清楚类属性 实例属性,就很容易写出来,原理完全一模一样. 如下: 源码: class A(object): d ...

  9. 03-python的新式类和经典类区别

    新式类就是  class person(object): 这种形式的, 从py2.2 开始出现的 新式类添加了: __name__ is the attribute's name. __doc__ i ...

  10. python中新式类和经典类

    python中的类分为新式类和经典类,具体有什么区别呢?简单的说, 1.新式类都从object继承,经典类不需要. Python 2.x中默认都是经典类,只有显式继承了object才是新式类 Pyth ...

随机推荐

  1. 分布式数据库中间件Mycat百亿级数据存储(转)

    此文转自: https://www.jianshu.com/p/9f1347ef75dd 2013年阿里的Cobar在社区使用过程中发现存在一些比较严重的问题,如高并发下的假死,心跳连接的故障,只实现 ...

  2. 前端获取URL和SESSON中的值

    .CS中代码 public ActionResult Index(string viewname, bool partial = false) { //获取URL中的 foreach (var key ...

  3. mysql数据库工具

    1.navicat12 中文版及破解 链接:https://pan.baidu.com/s/1TH8m6lduHJybUGhmjFPIAA 提取码:kwcd 2.旧版本mysql-front(连接可选 ...

  4. ncnblogs.com的用户体验

    你是什么样的用户, 有什么样的心理, 对cnblogs 的期望值是什么? 我是一名普通的学生,上cnblogs的期望是发表博客完成老师布置的任务. 当你第一次使用cnblogs 的功能的时候, 碰到了 ...

  5. whu Problem 1537 - A - Stones I 贪心

    题目链接: http://acm.whu.edu.cn/land/problem/detail?problem_id=1537 Stones I Time Limit: 1000MSMemory Li ...

  6. lintcode-383-装最多水的容器

    383-装最多水的容器 给定 n 个非负整数 a1, a2, ..., an, 每个数代表了坐标中的一个点 (i, ai).画 n 条垂直线,使得 i 垂直线的两个端点分别为(i, ai)和(i, 0 ...

  7. spring重定向与转发

    1.重定向 A,通过ModelAndView以及RedirectView @RequestMapping("/login1.do") public ModelAndView log ...

  8. Markdown使用github风格时报TLS错误解决办法

    https://docs.microsoft.com/en-us/officeonlineserver/enable-tls-1-1-and-tls-1-2-support-in-office-onl ...

  9. PHP对象的遍历

    对象的遍历 对象的遍历,跟数组的遍历,一样! 其实,只能遍历出对象的“实例属性数据” foreach( $对象名  as   $key => $value){ //这里就可以处理$key和$va ...

  10. perf的采样模式和统计模式

    perf的采样模式和统计模式 统计模式和采样模式使用寄存器的方法不相同; 在统计模式下,每次调度之前设置寄存器,调度之后清理寄存器,留个下个进程使用;PMU寄存器的使用方法; 在采样模式下,每次 pm ...