Python对象比较排序依赖魔术方法;需正确实现__eq__与__hash__一致性,避免修改状态,复杂对象优先用dataclass(order=True)自动生成。
Python中对象的比较和排序依赖于特定的魔术方法,正确实现这些方法能让自定义类支持==、、sorted()等操作,避免TypeError或逻辑错误。
当使用==比较两个对象时,Python会调用__eq__方法。默认行为是基于对象身份(is),即不同实例即使属性相同也返回False。要按内容判断相等,需显式实现__eq__,并通常同时实现__hash__(若需放入集合或作为字典键)。
__eq__中先检查类型:if not isinstance(other, type(self)): return False
True或False,不要返回其他值,否则可能引发意外行为__eq__但未定义__hash__,实例将自动变为不可哈希(hash(obj)报错)Python 3 中取消了__cmp__,改用六个富比较方法:__lt__(__le__(__eq__(==)、__ne__(!=)、__gt__(>)、__ge__(>=)。只要实现了其中一部分,其余可通过functools.total_ordering自动补全。
__eq__和一个如__lt__,其余比较逻辑自动推导total_ordering不会覆盖已定义的方法,适合渐进式补充sorted()、list.sort()等默认按升序排列,其底层调用的是对象的__lt__方法。只要类实现了__lt__(或通过total_ordering间接支持),就能直接参与排序。
sorted([obj1, obj2, obj3])会反复调用obj1.__lt__(obj2)等进行两两比较__lt__,会抛出TypeError: '
key参数绕过魔术方法,如sorted(items, key=lambda x: x.score)
忽略魔术方法的对称性或一致性,容易导致逻辑混乱。比如a == b为True,但hash(a) != hash(b)会使字典行为异常;又或a 和b > a结果不一致,影响排序稳定性。
__eq__时,确保相等对象的__hash__返回相同整数(若可哈希)dataclass(orde
r=True)自动生成比较方法,简洁且不易出错