前言
Python中的“is”和“==”用于对两个变量进行比较。
不同的是,“==”用于比较两个变量的值,而“is”用于比较两个变量的标识。
一、变量的理解
什么是变量?这是个不太好回答的问题。
在很多语言里,我们习惯性把变量理解成“存储数据的箱子(容器)”,然而这种理解方式不适用于引用式变量。因为两个变量可以指向同一个对象,但我们不能理解成两个箱子中分别存放着同一个数据,这是相当荒谬的。
在《流畅的Python》【巴西】Luciano Ramalho这本书中给了一个更加契合的解释:变量是附加在对象上的标注。
与把变量当成数据的容器不同,作者倾向于把引用式变量当做“便利贴”,一个物体上贴上多个“标签”,这是合理的。
二、“is”同一个数据的不同标签
有了上面的理解,我们来看一个例子:
>>> a = [1,2,3]
>>> b = a
>>> a is b
a和b是不同的两个变量名,但a和b标记的是同一个数据(List对象)。
Python中的每个对象都有一个唯一的标识,从被创建到被回收,伴随这个对象的“一生”。就像是人类的身份证一样,这个标识在变量的生命周期内是不会改变的。
而“is”就是比较两个变量所指对象的标识(ID)。这个标识,可以借助内置函数id()来获取。
换而言之,如果两个变量id相同,那么它们指向的是同一个对象。
关于id()函数,在不同的实现中返回的内容不同,例如CPython返回对象内存地址的整数表示,而其他实现Jython、IronPython等未必相同。但有一点可以确定,每个对象的id标识唯一。
三、“==”数据内容的比较
新建一个与a相同的列表:
>>> a = [1,2,3]
>>> b = a
>>> a is b
True
>>> id(a)
2478905612416
>>> id(b)
2478905612416
>>> c = [1,2,3]
>>> a is c
猜一猜“a is c”的返回结果是什么?没错,False。
虽然c所指的列表与a相等,但这两个对象却不相同。你是你,我是我,长得像而已,再没有其他关系。
“==”运算符,就是计算两个变量所指对象的值(内容)是否相同。
因为a和b指的是同一个对象,a == b的返回结果当然也是True。毕竟和我长得最像的,一定是我自己嘛。
四、“==”的实现
“==”运算是语法糖,可以理解为一种调用函数的简单书写方式。
当执行表达式“a == c”的时候,其实Python执行的是“【图放下面,markdown编辑器的文本样式打不出来】”这么一条语句。(eq是equal的缩写)
捎带问一句,如何打出编辑器本身用于样式的标记符号?
在Python中,大部分内置类型__eq__函数的实现都是比较两个变量所指对象的值,所以“==”的作用便成了判断值是否相等。
用户自定义的类,默认继承自object。但object的__eq__方法却是用于比较两个对象的ID。所以,如果你想让自定义类的实例可以正确使用“==”,需要自己编写__eq__函数进行覆盖。
“self.value == arg.value”会去调用内置类型Interger的__eq__函数。
总结
1、is用来判断两变量所指对象的ID是否相同,即是否为同一个对象。
2、==用来判断两变量所指对象的内容是否相同,即值是否相等。
3、==背后是在调用对象的__eq__方法,通过重写此方法,可以修改判断结果。