2

Python 网页上的文档中,python 中经典类的方法解析顺序被描述为深度优先从左到右搜索。我试图用这段代码对此进行测试:

class A(object):
def __init__(self):
    print "Initialized A"

class B(A):
    def test():
        print "Initialized B"

class C(A):
    def __init__(self):
        print "Initialized C"

class D(B, C):
    def __init__(self):
        super(D, self).__init__()
        print "Initialized D"

当我创建对象 D 的实例时:

D()

我得到结果:

Initialized C
Initialized D

虽然我期望打印“Initialized A”、“Initialized D”,因为 MRO 是深度优先的,但首先在 B 中搜索初始化,当未找到时(就是这种情况),它应该在层次结构中更深入并寻找B(即A)的基类中的函数。有人可以解释一下为什么我得到 C 的 init 函数而不是 A 的那个吗?

4

2 回答 2

5

MROD为:

print(D.mro())
[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <type 'object'>]

所以

super(D, self).__init__()

在after__init__的第一个类中寻找一个方法。由于没有,因此它会在 中查找方法。它发现,因此self.__class__.mro()DB__init____init__CC.__init__

super(D, self).__init__()

来电C.__init__(self)

请注意,super不是试图访问getattr(B, __init__). 相反,它正在寻找'__init__'in B.__dict__

于 2013-01-04T22:20:23.250 回答
4

您的类继承自object经典类,因此不是经典类。这是一门新式的课。(任何继承自新式类的类也是新式类。)

请注意,super无论如何都不能与经典类一起使用。您引用的“深度优先从左到右”规则只是当您尝试访问未在实例上直接定义的属性时搜索顺序基类的属性查找规则。经典类不提供这样的 MRO —— 如果不显式引用特定的超类,就无法访​​问属性的“超类值”。

于 2013-01-04T22:21:21.463 回答