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 相关文章推荐
wxpython 学习笔记 第一天
Feb 09 Python
跟老齐学Python之有点简约的元组
Sep 24 Python
python正则表达式re之compile函数解析
Oct 25 Python
Python实现爬虫从网络上下载文档的实例代码
Jun 13 Python
python微信公众号之关键词自动回复
Jun 15 Python
Python 实现取矩阵的部分列,保存为一个新的矩阵方法
Nov 14 Python
Django模型序列化返回自然主键值示例代码
Jun 12 Python
python GUI库图形界面开发之PyQt5单选按钮控件QRadioButton详细使用方法与实例
Feb 28 Python
Python 绘制可视化折线图
Jul 22 Python
Pycharm 如何一键加引号的方法步骤
Feb 05 Python
教你用python实现12306余票查询
Jun 30 Python
Python超详细分步解析随机漫步
Mar 17 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设计模式小结
2013/02/15 PHP
PHP基于工厂模式实现的计算器实例
2015/07/16 PHP
Javascript学习笔记2 函数
2010/01/11 Javascript
JQuery Ajax通过Handler访问外部XML数据的代码
2010/06/01 Javascript
Js基础学习资料
2010/11/23 Javascript
扩展Jquery插件处理mouseover时内部有子元素时发生样式闪烁
2011/12/08 Javascript
表单的焦点顺序tabindex和对应enter键提交
2013/01/04 Javascript
JavaScript创建对象的写法
2013/08/29 Javascript
JS实现局部选择打印和局部不选择打印
2014/04/03 Javascript
浅谈angularJS 作用域
2015/07/05 Javascript
基于javascript实现最简单的选项卡切换效果
2016/05/16 Javascript
深入理解jQuery之事件移除
2016/06/02 Javascript
node.js中 stream使用教程
2016/08/28 Javascript
微信小程序 swiper组件轮播图详解及实例
2016/11/16 Javascript
判断div滑动到底部的scroll实例代码
2017/11/15 Javascript
通过vue-router懒加载解决首次加载时资源过多导致的速度缓慢问题
2018/04/08 Javascript
Vue项目路由刷新的实现代码
2019/04/17 Javascript
简单实现节流函数和防抖函数过程解析
2019/10/08 Javascript
jquery html添加元素/删除元素操作实例详解
2020/05/20 jQuery
Python卸载模块的方法汇总
2016/06/07 Python
Python输出各行命令详解
2018/02/01 Python
Python 实现一行输入多个值的方法
2018/04/21 Python
详解Django的CSRF认证实现
2018/10/09 Python
浅谈python之高阶函数和匿名函数
2019/03/21 Python
python用win32gui遍历窗口并设置窗口位置的方法
2019/07/26 Python
Anaconda详细安装步骤图文教程
2020/11/12 Python
使用豆瓣源来安装python中的第三方库方法
2021/01/26 Python
利用SVG和CSS3来实现一个炫酷的边框动画
2015/07/22 HTML / CSS
2014年党务公开实施方案
2014/02/27 职场文书
葛优非诚勿扰搞笑征婚台词
2014/03/17 职场文书
质量承诺书格式范文
2015/04/28 职场文书
签约仪式致辞
2015/07/30 职场文书
2016寒假假期总结
2015/10/10 职场文书
Nginx同一个域名配置多个项目的实现方法
2021/03/31 Servers
深入理解python多线程编程
2021/04/18 Python
Python学习之包与模块详解
2022/03/19 Python