原因是如何查找特殊方法。
对于自定义类,只有在对对象的类型(而不是在对象的实例字典中)进行定义的情况下,才能保证对特殊方法的隐式调用可以正常工作。
…
此行为背后的原理在于许多特殊方法,例如__hash__()
和__repr__()
由所有对象(包括类型对象)实现。如果这些方法的隐式查找使用常规查找过程,则在对类型对象本身进行调用时它们将失败。
…
除了出于正确性的考虑绕过任何实例属性之外,隐式特殊方法查找通常还绕过该__getattribute__()
方法,即使对象的元类也是如此。
…
__getattribute__()
以这种方式绕过机器为解释器内的速度优化提供了很大的空间,但以牺牲一些特殊方法的灵活性为代价(特殊方法必须在类对象本身上设置,以便由解释器一致地调用) 。