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使用百度API上传文件到百度网盘代码分享
Nov 08 Python
python从入门到精通(DAY 2)
Dec 20 Python
python运行其他程序的实现方法
Jul 14 Python
python中MethodType方法介绍与使用示例
Aug 03 Python
Python判断变量名是否合法的方法示例
Jan 28 Python
Python应用领域和就业形势分析总结
May 14 Python
Python爬虫实现的根据分类爬取豆瓣电影信息功能示例
Sep 15 Python
python3中numpy函数tile的用法详解
Dec 04 Python
Python运行异常管理解决方案
Mar 09 Python
python中 _、__、__xx__()区别及使用场景
Jun 30 Python
OpenCV+Python3.5 简易手势识别的实现
Dec 21 Python
关于python中模块和重载的问题
Nov 02 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实现一维数组转二维数组的方法
2015/02/25 PHP
php使用Jpgraph绘制3D饼状图的方法
2015/06/10 PHP
分享3个php获取日历的函数
2015/09/25 PHP
Laravel中间件实现原理详解
2016/10/09 PHP
PHP7使用ODBC连接SQL Server2008 R2数据库示例【基于thinkPHP5.1框架】
2019/05/06 PHP
Javascript isArray 数组类型检测函数
2009/10/08 Javascript
jQuery UI 应用不同Theme的办法
2010/09/12 Javascript
js中访问html中iframe的文档对象的代码[IE6,IE7,IE8,FF]
2011/01/08 Javascript
myeclipse安装jQuery插件的方法
2011/03/29 Javascript
jQuery中:empty选择器用法实例
2014/12/30 Javascript
VueJS全面解析
2016/11/10 Javascript
AngularJS模仿Form表单提交的实现代码
2016/12/08 Javascript
bootstrap下拉菜单使用方法解析
2017/01/13 Javascript
JavaScript 通过Ajax 动态加载CheckBox复选框
2017/08/31 Javascript
vue实现购物车案例
2020/05/30 Javascript
javascript实现文字跑马灯效果
2020/06/18 Javascript
Python计时相关操作详解【time,datetime】
2017/05/26 Python
python如何使用正则表达式的前向、后向搜索及前向搜索否定模式详解
2017/11/08 Python
详解django.contirb.auth-认证
2018/07/16 Python
python re.sub()替换正则的匹配内容方法
2019/07/22 Python
Django框架之DRF 基于mixins来封装的视图详解
2019/07/23 Python
Python命令行click参数用法解析
2019/12/19 Python
Django中文件上传和文件访问微项目的方法
2020/04/27 Python
python 写一个水果忍者游戏
2021/01/13 Python
用CSS3打造HTML5的Logo(实现代码)
2016/06/16 HTML / CSS
德国滑雪和户外用品网上商店:XSPO
2019/10/30 全球购物
linux面试题参考答案(7)
2012/10/29 面试题
高中运动会广播稿
2014/01/21 职场文书
2014年三万活动总结
2014/04/26 职场文书
副科竞争上岗演讲稿
2014/05/12 职场文书
减负增效提质方案
2014/05/23 职场文书
运动会搞笑广播稿
2014/10/14 职场文书
先进基层党组织主要事迹材料
2015/11/03 职场文书
2019年年中职场激励人心语录30条
2019/08/07 职场文书
vue3语法糖内的defineProps及defineEmits
2022/04/14 Vue.js
python基础之//、/与%的区别详解
2022/06/10 Python