这在Python数据模型文档中进行了介绍:新型类的特殊方法查找:
对于新型类,只有在对对象的类型(而不是在对象的实例字典中)进行定义的情况下,才能保证对特殊方法的隐式调用可以正常工作。
和
此行为背后的原理在于许多特殊方法,例如__hash__()
和__repr__()
由所有对象(包括类型对象)实现。如果这些方法的隐式查找使用常规查找过程,则在对类型对象本身[。]进行调用时它们将失败。
所以,因为两者hash(int)
和hash(1)
必须的工作,特殊的方法查找的类型,而不是实例上。如果__hash__()
是在对象上抬头挺直,hash(int)
将被转换为int.__hash__()
,这将失败,因为int.__hash__()
是一个未绑定的方法,它预计将被称为上的实际情况 的int()
(如1
); 因此对于hash(int)
,type.__hash__()
应改为:
>>> hash(1) == int.__hash__(1)
True
>>> hash(int) == type.__hash__(int)
True
>>> int.__hash__()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: descriptor '__hash__' of 'int' object needs an argument
这是向后不兼容的更改,因此它仅适用于新型对象。