在Python中,经典类(class Person:)和新式类(class Person(object):)的主要区别就是体现在多继承的顺序上。
Python 2.x中默认都是经典类,只有显式继承了object才是新式类;
Python 3.x中默认都是新式类,不必显式地继承object。
假设现在有一个D类继承了B类和C类,B类和C类又分别继承了A类。
在Python2.7中分别执行如下经典类代码:
# Author:Zheng Na
# encoding=utf-8
#当D类中有构造函数时,实例化时会执行D类的构造函数
class A:
def __init__(self):
print("A") class B(A):
def __init__(self):
print("B") class C(A):
def __init__(self):
print("C") class D(B,C):
def __init__(self):
print("D") d = D() #输出:D
当D类中有构造函数时,实例化时会执行D类的构造函数
# Author:Zheng Na
# encoding=utf-8
#当D类中没有构造函数时,实例化时会执行B类的构造函数
class A:
def __init__(self):
print("A") class B(A):
def __init__(self):
print("B") class C(A):
def __init__(self):
print("C") class D(B,C):
pass d = D() #输出:B
当D类中没有构造函数时,实例化时会执行B类的构造函数
# Author:Zheng Na
# encoding=utf-8
#当D类、B类中没有构造函数时,实例化时会执行A类的构造函数
class A:
def __init__(self):
print("A") class B(A):
pass class C(A):
def __init__(self):
print("C") class D(B,C):
pass d = D() #输出:A
当D类、B类中没有构造函数时,实例化时会执行A类的构造函数
# Author:Zheng Na
# encoding=utf-8
#当D类、B类、A类中没有构造函数时,实例化时会执行C类的构造函数
class A:
pass class B(A):
pass class C(A):
def __init__(self):
print("C") class D(B,C):
pass d = D() #输出:C
当D类、B类、A类中没有构造函数时,实例化时会执行C类的构造函数
你会发现,Python2.7中经典类的继承顺序是深度优先策略,先一级一级的往上查找,查到根部没有的话,再返回下一级查找。
在Python2.7中分别执行如下新式类代码:
# Author:Zheng Na
# encoding=utf-8
# 当D类中有构造函数时,实例化时会执行D类的构造函数
class A(object):
def __init__(self):
print("A") class B(A):
def __init__(self):
print("B") class C(A):
def __init__(self):
print("C") class D(B, C):
def __init__(self):
print("D") d = D() # 输出:D
当D类中有构造函数时,实例化时会执行D类的构造函数
# Author:Zheng Na
# encoding=utf-8
# 当D类中没有构造函数时,实例化时会执行B类的构造函数
class A(object):
def __init__(self):
print("A") class B(A):
def __init__(self):
print("B") class C(A):
def __init__(self):
print("C") class D(B, C):
pass d = D() # 输出:B
当D类中没有构造函数时,实例化时会执行B类的构造函数
# Author:Zheng Na
# encoding=utf-8
# 当D类、B类中没有构造函数时,实例化时会执行C类的构造函数
class A(object):
def __init__(self):
print("A") class B(A):
pass class C(A):
def __init__(self):
print("C") class D(B, C):
pass d = D() # 输出:C
当D类、B类中没有构造函数时,实例化时会执行C类的构造函数
# Author:Zheng Na
# encoding=utf-8
# 当D类、B类、C类中没有构造函数时,实例化时会执行A类的构造函数
class A(object):
def __init__(self):
print("A") class B(A):
pass class C(A):
pass class D(B, C):
pass d = D() # 输出:A
当D类、B类、C类中没有构造函数时,实例化时会执行A类的构造函数
你会发现,Python2.7中新式类的继承顺序是广度优先策略,先在上一级从左到右查找一遍,没有的话,再到上上一级从左到右查找一遍,以此类推。
经典类 | 新式类 | |
Python2.x | 深度优先 | 广度优先 |
Python3.x | 广度优先 | 广度优先 |
Python2.x的经典类是按照深度优先策略来继承的,新式类是按照广度优先策略来继承的。
Python3.x的经典类和新式类统一都是按照广度优先策略来继承的。
这并不是Python 2.x中的一个bug,只能算是一种策略。由于在大多数情况下,深度优先的效率没有广度优先高,所以在Python 3.x中才全部改为了广度优先。
另外,super()也是新式类的用法。