经常在代码中发现有时候用is,而有时候用==,弄不清楚两者到底有何区别,今天闲下来弄清楚
首先让我们来看下面的几个例子:
a = "hello"
b = "hello"
print(a is b) #输出True
print(a==b) #输出True
a = "hello world"
b = "hello world"
print(a is b) #输出Flase
print(a==b) #输出True
a = [1, 2, 3]
b = [1, 2, 3]
print(a is b) #输出False
print(a==b) #输出True
a = [1, 2, 3]
b = a
print(a is b) #输出True
print(a==b) #输出True
上面的输出结果,为什么有时 is 和 == 输出结果相同,而有时不同呢?我们来看看官方文档对 is 和 == 的解释。
官方文档中说 is 表示对象标识符(object identity),而 == 表示相等(equality)。is 的作用是检查对象的标识符是否相等,也就是比较对象在内存中的地址是否一样;而 == 是检查两个对象是否相等。
我们在检查 a is b 的时候,其实相当于检查 id(a) == id(b)。而检查 a == b 的时候,实际是调用了对象 a 的 __eq()__ 方法,a == b 相当于 a.__eq__(b)。一般情况下,如果 a is b 返回True的话,即 a 和 b 指向同一块内存地址,a == b 也返回True,即 a 和 b 的值也相等。
看明白上面的解释之后,我们再来看几个例子:
a = "hello"
b = "hello"
print(id(a)) # 输出 140506224367496
print(id(b)) # 输出 140506224367496
print(a is b) # 输出 True
print(a == b) # 输出 True
a = "hello world"
b = "hello world"
print(id(a)) # 输出 140506208811952
print(id(b)) # 输出 140506208812208
print(a is b) # 输出 False
print(a == b) # 输出 True
a = [1, 2, 3]
b = [1, 2, 3]
print(id(a)) # 输出 140506224299464
print(id(b)) # 输出 140506224309576
print(a is b) # 输出 False
print(a == b) # 输出 True
a = [1, 2, 3]
b = a
print(id(a)) # 输出 140506224305672
print(id(b)) # 输出 140506224305672
print(a is b) # 输出 True
print(a == b) # 输出 True
打印出 id(a) 和 id(b) 后就很清楚了。只要 a 和 b 的值相等,a == b 就会返回True,而只有 id(a) 和 id(b) 相等时,a is b 才返回 True。
这里还有一个问题,为什么 a 和 b 都是 “hello” 的时候,a is b 返回True,而 a 和 b都是 “hello world” 的时候,a is b 返回False呢?
这是因为前一种情况下Python的字符串驻留机制起了作用。对于只包含字母、数字、下划线的字符串,为了提高系统性能Python会保留其值的一个副本,当创建新的字符串的时候直接指向该副本即可。所以 “hello” 在内存中只有一个副本,a 和 b 的 id 值相同,而 “hello world” 这个字符串包含空格字符,不驻留内存,Python中各自创建了对象来表示 a 和 b,所以他们的值相同但 id 值不同。
总结一下,is 是检查两个对象是否指向同一块内存空间,而 == 是检查他们的值是否相等。可以看出,is 是比 == 更严格的检查,is 返回True表明这两个对象指向同一块内存,值也一定相同。