if foobar != None 和 if foobar is not None 是完全等价的吗?

2018-03-31 12:11:48 +08:00
 miniyao

有两个基础的问题,都有用过,不确定哪些场景完全等价 /不等价:

1. if foobar != None 和 if foobar is not None 是完全等价吗?

2. if foobar is not None 和 if foobar 是完全等价吗?
3250 次点击
所在节点    Python
12 条回复
jingniao
2018-03-31 12:15:58 +08:00
都不等价
lyc8801
2018-03-31 12:18:41 +08:00
不等,is,is not 好像是用来判断引用的对象是不是同一个,==,!=这种是判断值的
zeyexe
2018-03-31 12:23:56 +08:00
1. 参考 https://stackoverflow.com/a/1504742
2. if foobar 是判断真值,foobar=0 的时候 if foobar 也是 False
iEverX
2018-03-31 12:33:30 +08:00
@lyc8801 对于 None 来说,==和 is 应该没差别,语意上可能不一样,结果上应该是恒等的。
binux
2018-03-31 13:22:03 +08:00
@iEverX #4 重载 __eq__
gwki
2018-03-31 13:23:07 +08:00
先弄清 value 和 identity
value 可能会变,多个对象可以拥有相同的 value
identity 自对象创建到其被销毁都不会变,每一个对象的 identity 都不同,id(x)可以返回 x 对象的 identity
CPython 将 id(x)实现为返回存储对象的内存地址

is 测试两边运算对象的 id 是否相同,当且仅当对象 x 和 y 的 id(x) == id(y)时 is 返回 True
is not 返回与 is 相反的结果

!=, ==, >, <等叫做值比较,而值比较的默认行为可以通过修改__eq__,__ne__等函数来改变

if 在一个对象的__bool__()不返回 False 和__len__()不返回 0 的情况下认定对象为真,还有,语言内置的常量 None 和 False 被认为是假的

1. if foobar != None 和 if foobar is not None 是完全等价吗?
在下面这种或与其类似的情况下是完全等价的:
class Hello:
def __ne__(self, other):
print("Excalibur !")
return (self is not other)

foobar = Hello()
print(foobar is not None)
print(foobar != None)
一般而言,是不完全等价的
因为 is not 测试 foobar 的 id 和 None 的 id 是否相同,而!=测试 foobar 的值和 None 的值是否相同

2. if foobar is not None 和 if foobar 是完全等价吗?
一般情况下是不等价的,因为前者测试 id 是否相同,后者测试 foobar 的__bool__()或__len__()
当然你也可以修改 foobar 的__bool__函数
以上结论基于 Python3.6 的官方文档^^/
iEverX
2018-03-31 13:29:46 +08:00
@binux 说的是
scriptB0y
2018-03-31 14:38:23 +08:00
楼上 @gwki 说的很清楚了!

但是对于 None 来说有一点区别,你看很多 Python 代码就会发现:大部分情况下我们用 if foo is None 来做判断,因为 None 在 Python 中是一个全局唯一变量。官方文档中说:Since None is a singleton, testing for object identity (using == in C) is sufficient. 所以官方是推荐用 id 来 check 的。

即:None 只有一个,不存在值为 None 但是与 id(None) 不相等的情况。

写作 if foo != None 有点不 Pythonic (反正我是没这么见过哈哈哈)。

问题 2:

foo = 0
if foo 判断为假,
if foo is not None 判断为真。所以 is 判断的是 id 相同(对于 None 来说判断 id 相同和判断值相同没有太大区别,反正只有 1 个)。

所以二者是不一样的,除了 None 之外,文档( https://docs.python.org/3.6/library/stdtypes.html#truth-value-testing )还有下面的判断为假:

- constants defined to be false: None and False.
- zero of any numeric type: 0, 0.0, 0j, Decimal(0), Fraction(0, 1)
- empty sequences and collections: '', (), [], {}, set(), range(0)

再啰嗦一点,对于不可变对象,为了避免重复创建,Python 做了驻留处理。比如下面代码:

>>> s1 = "ABC"
>>> s2 = "ABC"
>>> s1 is s2
True

但是我们实际比较二者的时候,应该用 s1 == s2。因为驻留操作是 CPython 的实现细节。副作用不应该被依赖。
scriptB0y
2018-03-31 14:40:22 +08:00
贴两篇博客,有时间可以参考下:

《详解 Python 的 “==” 和 “ is ”》 https://www.kawabangga.com/posts/1673
《作用还是 Feature ?》 https://www.kawabangga.com/posts/2809
vimiix
2018-03-31 15:27:42 +08:00
延伸阅读,True,1,1.0 分别作为字典的值的时候会怎样。
vimiix
2018-03-31 15:28:02 +08:00
fanhaipeng0403
2018-03-31 22:34:13 +08:00
印象中 None null 这样的类型

不能用=

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/443096

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX