Python判断两个对象相等的原理


Posted in Python onDecember 12, 2017

概述

大部分的python程序员平时编程的时候,很少关心两个对象为什么相等,因为教程和经验来说,他们就应该相等,比如1==1就应该返回True,可是当我们想要定义自己的对象或者修改默认的对象行为时,通常会因为不了解原理而导致各种奇奇怪怪的错误。

两个对象如何相等

两个对象如何才能相等要比我们想象的复杂很多,但核心的方法是重写 eq 方法,这个方法返回True,则表示两个对象相等,否则,就不相等。相反的,如果两个对象不相等,则重写 ne 方法。 默认情况下,如果你没有实现这个方法,则使用父类(object)的方法。父类的方法比较是的两个对象的ID(可以通过id方法获取对象ID),也就是说,如果对象的ID相等,则两个对象也就相等。因此,我们可以得知,默认情况下,对象只和自己相等。例如:

>>> class A(object):
...  pass
...
>>>
>>> a = A()
>>> b = A()
>>> a == a
True
>>> a == b
False
>>> id(a)
4343310992
>>> id(b)
4343310928

Python2程序员经常犯的一个错误是,只重写了 eq 方法,而没有重写 ne 方法,导致不可预计的错误。而Python3会自动重写 ne 方法,如果你没有重写的话。

对象的Hash方法

Python里可Hash的对象,都有一个数字ID代表了它在python里的值,这个ID是由对象的 hash 方法返回的。因此,如果想让一个对象可Hash,那必须实现 hash 方法和之前提到的 eq 方法。和对象相等一样,默认情况下,对象的 hash 方法继承自Object对象,而Object对象的 hash 方法只计算对象ID,因此两个对象始终拥有两个不一样的hash id,不管他们是多么相似。 当我们把一个不可Hash的对象加入到set或者dict时,会发生什么了?

>>> set().add({})
Traceback (most recent call last):
 File "<stdin>", line 1, in <module>
TypeError: unhashable type: 'dict'
unhashable type: 'dict'

原因是set()和dict()使用对象的hash值作为内部索引,以便能快速索引到指定对象。因此,同一个对象返回相同的hash id就很重要了。

对象的Hash值在它的生命周期内不能改变

如果你想定义一个比较完美的对象,并且实现了 eq 和 hash 方法来定义对象的比较行为和hash值,那么你就需要保证对象的相关属性不能发生更改。不然会导致很诡异的错误,比如下面的例子。

>>> class C:
...  def __init__(self, x):
...   self.x = x
...  def __repr__(self):
...   return "C({"+str(self.x)+"})"
...  def __hash__(self):
...   return hash(self.x)
...  def __eq__(self, other):
...   return (
...    self.__class__ == other.__class__ and
...    self.x == other.x
...   )
>>> d = dict()
>>> s = set()
>>> c = C(1)
>>> d[c] = 42
>>> s.add(c)
>>> d, s
({C(1): 42}, {C(1)})
>>> c in s and c in d # c is in both!
True
>>> c.x = 2
>>> c in s or c in d # c is in neither!?
False
>>> d, s
({C(2): 42}, {C(2)}) # but...it's right there!

在我们没有修改对象的属性时(c.x=2)之前,所有行为都符合预期。当我们通过c.x=2时修改属性后,执行c in s or c in d返回False,但是内容却是修改后的,是不是很奇怪。这也就解释了为什么str、tuple是可Hash的,而list和dict是不可hash的。

因此我们可以得出结论,如果两个对象相等的话,那它们的hash值必然也是相等的。

总结

讲了这么多有什么用了。 1. 当我们遇到unhashable type这个异常时,我们能够知道为什么报这个错误。 2. 如果定义了一个可比较的对象,那么最好保证对象hash值相关的属性在生命周期内不能发生改变,不然会发生意想不到的错误。

以上所述是小编给大家介绍的Python判断两个对象相等的原理,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Python 相关文章推荐
python查看微信好友是否删除自己
Dec 19 Python
使用Python绘制图表大全总结
Feb 11 Python
Python模拟登陆实现代码
Jun 14 Python
Python3 循环语句(for、while、break、range等)
Nov 20 Python
Python反射的用法实例分析
Feb 11 Python
PyQt5每天必学之布局管理
Apr 19 Python
python批量复制图片到另一个文件夹
Sep 17 Python
Python3.5内置模块之time与datetime模块用法实例分析
Apr 27 Python
matplotlib subplot绘制多个子图的方法示例
Jul 28 Python
python 如何在测试中使用 Mock
Mar 01 Python
python批量创建变量并赋值操作
Jun 03 Python
利用Python多线程实现图片下载器
Mar 25 Python
浅谈Django REST Framework限速
Dec 12 #Python
Django admin美化插件suit使用示例
Dec 12 #Python
Django admin实现图书管理系统菜鸟级教程完整实例
Dec 12 #Python
基于Django filter中用contains和icontains的区别(详解)
Dec 12 #Python
Python有序查找算法之二分法实例分析
Dec 11 #Python
django实现用户登陆功能详解
Dec 11 #Python
Python通过Django实现用户注册和邮箱验证功能代码
Dec 11 #Python
You might like
解析thinkphp中的M()与D()方法的区别
2013/06/22 PHP
php颜色转换函数hex-rgb(将十六进制格式转成十进制格式)
2013/09/23 PHP
PDO防注入原理分析以及注意事项
2015/02/25 PHP
codeigniter发送邮件并打印调试信息的方法
2015/03/21 PHP
php文件操作相关类实例
2015/06/18 PHP
ThinkPHP在Cli模式下使用模板引擎的方法
2015/09/25 PHP
基于PHP实现通过照片获取ip地址
2016/04/26 PHP
功能强大的php文件上传类
2016/08/29 PHP
PHP数组操作简单案例分析
2016/10/15 PHP
高性能WEB开发 flush让页面分块,逐步呈现 flush让页面分块,逐步呈现
2010/06/19 Javascript
javascript实现的一个带下拉框功能的文本框
2014/05/08 Javascript
JS实现文件动态顺序载入的方法
2015/03/07 Javascript
DIV随滚动条滚动而滚动的实现代码【推荐】
2016/04/12 Javascript
轮播的简单实现方法
2016/07/28 Javascript
关于javascript的一些知识以及循环详解
2016/09/12 Javascript
js实现选项卡内容切换以及折叠和展开效果【推荐】
2017/01/08 Javascript
使用 vue.js 构建大型单页应用
2018/02/10 Javascript
JS中promise化微信小程序api
2018/04/12 Javascript
vue2单元测试环境搭建
2018/05/24 Javascript
js屏蔽退格键(backspace或者叫后退键与F5)
2019/02/10 Javascript
js实现简单的贪吃蛇游戏
2020/04/23 Javascript
小程序实现tab标签页
2020/11/16 Javascript
python实现的解析crontab配置文件代码
2014/06/30 Python
Python的设计模式编程入门指南
2015/04/02 Python
Django自定义过滤器定义与用法示例
2018/03/22 Python
python实现内存监控系统
2021/03/07 Python
Python3.5 处理文本txt,删除不需要的行方法
2018/12/10 Python
Python实现高斯函数的三维显示方法
2018/12/29 Python
python实现两个经纬度点之间的距离和方位角的方法
2019/07/05 Python
python 利用pyttsx3文字转语音过程详解
2019/09/25 Python
Pytorch生成随机数Tensor的方法汇总
2020/09/09 Python
团工委书记自荐书范文
2013/12/17 职场文书
广告创意求职信
2014/03/17 职场文书
维护民族团结演讲稿
2014/08/27 职场文书
镇人大副主席民主生活会对照检查材料思想汇报
2014/10/01 职场文书
MySQL数据库索引的最左匹配原则
2021/11/20 MySQL