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求素数示例分享
Feb 16 Python
python基本语法练习实例
Sep 19 Python
Python数据结构与算法之二叉树结构定义与遍历方法详解
Dec 12 Python
Python爬虫实现百度图片自动下载
Feb 04 Python
详解Python中如何写控制台进度条的整理
Mar 07 Python
python执行系统命令后获取返回值的几种方式集合
May 12 Python
Python自动抢红包教程详解
Jun 11 Python
python中bytes和str类型的区别
Oct 21 Python
Python实现疫情通定时自动填写功能(附代码)
May 27 Python
Python LMDB库的使用示例
Feb 14 Python
Python使用scapy模块发包收包
May 07 Python
Python数据结构之队列详解
Mar 21 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编程开发“虚拟域名”系统
2006/10/09 PHP
php tp验证表单与自动填充函数代码
2012/02/22 PHP
ThinkPHP之getField详解
2014/06/20 PHP
thinkPHP统计排行与分页显示功能示例
2016/12/02 PHP
鼠标滚轮控制网页横向移动实现思路
2013/03/22 Javascript
基于JavaScript 类的使用详解
2013/05/07 Javascript
web网页按比例显示图片实现原理及js代码
2013/08/09 Javascript
给事件响应函数传参数的四种方式小结
2013/12/05 Javascript
jquery自动切换tabs选项卡的具体实现
2013/12/24 Javascript
基于jquery的文字向上跑动类似跑马灯的效果
2014/09/22 Javascript
JS和JQ的event对象区别分析
2014/11/24 Javascript
Javascript 高阶函数使用介绍
2015/06/15 Javascript
深入分析下javascript中的[]()+!
2015/07/07 Javascript
Javascript连接Access数据库完整实例
2015/08/03 Javascript
AngularJS中比较两个数组是否相同
2016/08/24 Javascript
jQuery插件FusionCharts绘制的3D双柱状图效果示例【附demo源码】
2017/04/20 jQuery
微信小程序按钮去除边框线分享页面功能
2018/08/27 Javascript
vue.js编译时给生成的文件增加版本号
2018/09/17 Javascript
关于layui时间回显问题的解决方法
2019/09/24 Javascript
jQuery实时统计输入框字数及限制
2020/06/24 jQuery
vue+ElementUI 关闭对话框清空验证,清除form表单的操作
2020/08/06 Javascript
Python中使用select模块实现非阻塞的IO
2015/02/03 Python
pygame播放音乐的方法
2015/05/19 Python
django通过ajax发起请求返回JSON格式数据的方法
2015/06/04 Python
Python多进程分块读取超大文件的方法
2016/04/13 Python
详解Python在七牛云平台的应用(一)
2017/12/05 Python
详解Python异常处理中的Finally else的功能
2017/12/29 Python
Python定时任务APScheduler的实例实例详解
2019/07/22 Python
15个应该掌握的Jupyter Notebook使用技巧(小结)
2020/09/23 Python
python中re模块知识点总结
2021/01/17 Python
html5菜单折纸效果
2014/04/22 HTML / CSS
New Balance天猫官方旗舰店:始于1906年,百年慢跑品牌
2017/11/15 全球购物
Opodo英国旅游网站:预订廉价航班、酒店和汽车租赁
2018/07/14 全球购物
幼儿园教学管理制度
2014/02/04 职场文书
中国在我心中演讲稿
2014/09/13 职场文书
加强作风建设演讲稿
2014/10/24 职场文书