Python | Python 的 is 和 == 你了解么?

Python 是一门面向对象的语言,Python中一切皆对象。

Python 中对象包含的三个基本要素,分别是:

  1. id(身份标识)
  2. type(数据类型)
  3. value(值)

其中 id 也代表着内存地址,Python 对象之间比较是否相等既可以用 == ,也可以用 is ,那么二者的区别在哪呢?

既然说到 id 也代表内存地址,那其实 is 比较的就是对象的内存地址,如果两个对象的 id 相等,则说明这两个对象指向的是同一个内存地址,是同一实例对象。而 == 就是比较两个对象的值是否相等,当然,前期是这二者的数据类型一致,否则根本无可比性。另外,== 比较对象通常会调用 Python 的 _ eq _ 方法。

先上一波代码来解释一下。

# partA
>>> j = [6,6,6,6,6,6]
>>> b = j
>>> print(b == j)
True
>>> print(b is j)
True
# partB
>>> j = [6,6,6,6,6,6]
>>> b = [6,6,6,6,6,6]
>>> print(b == a)
True
>>> print(b is a)
False

留意代码你可能有疑问,为毛同样是 list 类型,值一样是 [6,6,6,6,6,6] ,partA 和 partB 的结果不一样呢?

原因是 partA 中,代码 b = j ,意指把 j 赋值给 b,其实底层做得就是将 b 指向 j 的内存地址。而此时二者 id 相等,值相等。而 partB 中,虽然二者的值都是一样的,但是此时 b 并不是指向 j ,而是在内存中重新开辟新空间存储 [6,6,6,6,6,6]。所以 二者的 id 不等,值相等。

不信?你看

# partA
>>> j = [6,6,6,6,6,6]
>>> b = j
>>> print(b == j)
True
>>> print(b is j)
True

>>> print(id(j))
1664043697800
>>> print(id(b))
1664043697800

#partB
>>> j = [6,6,6,6,6,6]
>>> b = [6,6,6,6,6,6]
>>> print(b == a)
True
>>> print(b is a)
False

>>> print(id(j))
1664043698376
>>> print(id(b))
1664043697864

这下该信了吧?

一种特殊情况,先看代码:

# partA
>>> x = 256
>>> y = 256
>>> print(x == y)
True
>>> print(x is y)
True

# partB
>>> x = 1000
>>> y = 1000
>>> print(x == y)
True
>>> print(x is y)
False

我估计又有朋友要惊了个呆了。为毛 x,y 都等于 256 时,is 和 == 的结果是一样的,而 x,y 都等于 1000 时,is 就叛变了呢?

其实这要说到 Python 的优化了。Python 为了性能做了许多优化。而其中就包括对较小的,常用的整数对象的优化,Python 底层有一个名叫 small_ints 的链表。这个链表存储了 Pyhton 最常用的一些小整数,而这些整数的范围就在 [-5 , 256] 之间。每当 程序需要用到这个范围内的整数, Python 就直接到底层链表去取,而不需要重新创建一个整数对象。

比较字符串

# partA
>>> n = 'nasus666'
>>> m = 'nasus666'
>>> print(m == n)
True
>>> print(m is n)
True

# partB
>>> m = 'nasus.666'
>>> n = 'nasus.666'
>>> print(m == n)
True
>>> print(m is n)
False

字符串类型不完全相同,这个结果应该和解释器实现有关。(此处具体原因不明觉厉,请知道的朋友后台留言,万分感谢)

各数据类型的比较结果

# 元组
>>> x = (6,6,6)
>>> y = (6,6,6)
>>> print(x is y)
False
>>> print(x == y)
True

# list
>>> x = [6,6,6]
>>> y = [6,6,6]
>>> print(x is y)
False
>>> print(x == y)
True

# dict 
>>> x = {'狗哥':'很有钱', '家产':'过亿'}
>>> y = {'狗哥':'很有钱', '家产':'过亿'}
>>> print(x == y)
True
>>> print(x is y)
False

# set
>>> x = ([6,6,6,6,6])
>>> y = ([6,6,6,6,6])
>>> print(x == y)
True
>>> print(x is y)
False

当对象为数字,字符串,list,dict,set时,is 和 == 的结果都不同。所以不能随便交换 is 和 == 去比较对象。is 比较的是对像的内存地址,id 值。而 == 更多的是比较对象的值。

总结

  • is 比较两个对象的 id 值是否相等,是否指向同一个内存地址;
  • == 比较的是两个对象的内容是否相等,值是否相等;
  • 小整数对象 [-5,256] 在全局解释器范围内被放入缓存供重复使用;
  • is 运算符比 == 效率高,在变量和 None 进行比较时,应该使用 is。

本文分享自微信公众号 - 一个优秀的废人(feiren_java)

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2018-07-28

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

扫码关注云+社区

领取腾讯云代金券