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 相关文章推荐
Python3实现连接SQLite数据库的方法
Aug 23 Python
python的Template使用指南
Sep 11 Python
python打开文件并获取文件相关属性的方法
Apr 23 Python
python实现按任意键继续执行程序
Dec 30 Python
Python中的命令行参数解析工具之docopt详解
Mar 27 Python
Python编写Windows Service服务程序
Jan 04 Python
Python实现识别手写数字 Python图片读入与处理
Mar 23 Python
Python callable()函数用法实例分析
Mar 17 Python
Python设计模式之备忘录模式原理与用法详解
Jan 15 Python
快速解决vue.js 模板和jinja 模板冲突的问题
Jul 26 Python
Python 实现的 Google 批量翻译功能
Aug 26 Python
快速查找Python安装路径方法
Feb 06 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
世界上第一台立体声收音机
2021/03/01 无线电
php 数组的一个悲剧?
2011/05/11 PHP
PHP中使用addslashes函数转义的安全性原理分析
2014/11/03 PHP
PHP获取文件夹内文件数的方法
2015/03/12 PHP
Highslide.js是一款基于js实现的网页中图片展示插件
2020/03/30 Javascript
Blocksit插件实现瀑布流数据无限( 异步)加载
2014/06/20 Javascript
js单独获取一个checkbox看其是否被选中
2014/09/22 Javascript
JavaScript的内存释放问题详解
2015/01/21 Javascript
js中日期的加减法
2015/05/06 Javascript
javascript发送短信验证码实现代码
2015/11/12 Javascript
基于jQuery Ajax实现上传文件
2016/03/24 Javascript
利用css+原生js制作简单的钟表
2020/04/07 Javascript
详解js的事件处理函数和动态创建html标记方法
2016/12/16 Javascript
JavaScript cookie详解及简单实例应用
2016/12/31 Javascript
canvas实现图像布局填充功能
2017/02/06 Javascript
jQuery制作全屏宽度固定高度轮播图(实例讲解)
2017/07/08 jQuery
解决jquery的ajax调取后端数据成功却渲染失败的问题
2018/08/08 jQuery
解决vue项目使用font-awesome,build后路径的问题
2018/09/01 Javascript
如何在 JavaScript 中更好地利用数组
2018/09/27 Javascript
微信小程序事件 bindtap bindinput代码实例
2019/08/26 Javascript
vue 获取元素额外生成的data-v-xxx操作
2020/09/09 Javascript
[01:01:42]Secret vs Optic Supermajor 胜者组 BO3 第二场 6.4
2018/06/05 DOTA
[48:47]VGJ.S vs NB 2018国际邀请赛小组赛BO2 第一场 8.18
2018/08/19 DOTA
python检查字符串是否是正确ISBN的方法
2015/07/11 Python
如何使用Python进行PDF图片识别OCR
2021/01/22 Python
美国婴儿服装购物网站:Gerber Childrenswear
2020/05/06 全球购物
2013年办公室秘书的个人自我鉴定
2013/10/24 职场文书
简短证婚人证婚词
2014/01/09 职场文书
迅雷Cued工作心得体会
2014/01/27 职场文书
初中班主任评语大全
2014/04/24 职场文书
工地安全质量标语
2014/06/07 职场文书
卖车协议书范例
2014/09/16 职场文书
乡镇安全生产月活动总结
2015/05/08 职场文书
小学六一主持词开场白
2015/05/28 职场文书
优秀新员工事迹材料
2019/05/13 职场文书
解析高可用Redis服务架构分析与搭建方案
2021/06/20 Redis