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 def函数的定义、使用及参数传递实现代码
Aug 10 Python
浅析Python中元祖、列表和字典的区别
Aug 17 Python
python中requests库session对象的妙用详解
Oct 30 Python
python实现媒体播放器功能
Feb 11 Python
python用pandas数据加载、存储与文件格式的实例
Dec 07 Python
基于Python实现ComicReaper漫画自动爬取脚本过程解析
Nov 11 Python
python GUI库图形界面开发之PyQt5菜单栏控件QMenuBar的详细使用方法与实例
Feb 28 Python
Pycharm激活码激活两种快速方式(附最新激活码和插件)
Mar 12 Python
python3.7+selenium模拟淘宝登录功能的实现
May 26 Python
python实现学生成绩测评系统
Jun 22 Python
Python 中数组和数字相乘时的注意事项说明
May 10 Python
OpenCV-Python实现轮廓的特征值
Jun 09 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
php数据库密码的找回的步骤
2011/01/12 PHP
php读取txt文件组成SQL并插入数据库的代码(原创自Zjmainstay)
2012/07/31 PHP
浅谈使用PHP开发微信支付的流程
2015/10/04 PHP
Javascript中暂停功能的实现代码
2007/03/04 Javascript
javascript入门·对象属性方法大总结
2007/10/01 Javascript
关于viewport,Ext.panel和Ext.form.panel的关系
2009/05/07 Javascript
jQuery登陆判断简单实现代码
2013/04/21 Javascript
js图片延迟加载的实现方法及思路
2013/07/22 Javascript
jquery学习总结(超级详细)
2014/09/04 Javascript
使用angular写一个hello world
2015/01/23 Javascript
Javascript中的getUTCDay()方法使用详解
2015/06/10 Javascript
jquery+ajaxform+springboot控件实现数据更新功能
2018/01/22 jQuery
教你如何用node连接redis的示例代码
2018/07/12 Javascript
vue-cli监听组件加载完成的方法
2018/09/07 Javascript
详解小程序设置缓存并且不覆盖原有数据
2019/04/15 Javascript
[02:00]最后,我终于出了辉耀
2018/03/27 DOTA
Python 除法小技巧
2008/09/06 Python
python字符串替换示例
2014/04/24 Python
Python+Turtle动态绘制一棵树实例分享
2018/01/16 Python
Python 查看list中是否含有某元素的方法
2018/06/27 Python
python实现指定文件夹下的指定文件移动到指定位置
2018/09/17 Python
Python实现DDos攻击实例详解
2019/02/02 Python
浅谈Scrapy网络爬虫框架的工作原理和数据采集
2019/02/07 Python
对django views中 request, response的常用操作详解
2019/07/17 Python
Python字典中的值为列表或字典的构造实例
2019/12/16 Python
使用 Python 处理3万多条数据只要几秒钟
2020/01/19 Python
python字符串,元组,列表,字典互转代码实例详解
2020/02/14 Python
使用pth文件添加Python环境变量方式
2020/05/26 Python
利用python汇总统计多张Excel
2020/09/22 Python
德国机车企业:FC-Moto
2017/10/27 全球购物
第二层交换机和路由器的区别?第三层交换机和路由器的区别?
2013/05/23 面试题
如何写早恋检讨书
2014/09/10 职场文书
店铺转让协议书(2014版)
2014/09/23 职场文书
收入及婚姻状况证明
2014/11/20 职场文书
联谊活动总结范文
2015/05/09 职场文书
求职信如何撰写?
2019/05/22 职场文书