请选择 进入手机版|继续访问电脑版
发新帖

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

[复制链接]
4888 12

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有帐号?立即注册

x
有两个基础的问题,都有用过,不确定哪些场景完全等价 /不等价:
1. if foobar != None 和 if foobar is not None 是完全等价吗?2. if foobar is not None 和 if foobar 是完全等价吗?

举报 使用道具

回复

精彩评论12

jingniao  注册会员  发表于 2018-4-3 13:02:31 | 显示全部楼层
都不等价

举报 使用道具

回复
lyc8801  新手上路  发表于 2018-4-3 13:08:16 | 显示全部楼层
不等,is,is not 好像是用来判断引用的对象是不是同一个,==,!=这种是判断值的

举报 使用道具

回复
zeyexe  新手上路  发表于 2018-4-3 13:10:00 | 显示全部楼层
1. 参考 https://stackoverflow.com/a/1504742
2. if foobar 是判断真值,foobar=0 的时候 if foobar 也是 False

举报 使用道具

回复
ieverx  新手上路  发表于 2018-4-3 13:12:33 | 显示全部楼层
@lyc8801  对于 None 来说,==和 is 应该没差别,语意上可能不一样,结果上应该是恒等的。

举报 使用道具

回复
binux  新手上路  发表于 2018-4-3 13:15:49 | 显示全部楼层
@iEverX #4 重载 __eq__

举报 使用道具

回复
gwki  新手上路  发表于 2018-4-3 13:19:57 | 显示全部楼层
先弄清 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-4-3 13:24:42 | 显示全部楼层
@binux 说的是

举报 使用道具

回复
scriptb0y  注册会员  发表于 2018-4-3 13:39:40 | 显示全部楼层
楼上 @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-4-3 13:56:33 | 显示全部楼层
贴两篇博客,有时间可以参考下:

《详解 Python 的 “==” 和 “ is ”》 https://www.kawabangga.com/posts/1673
《作用还是 Feature ?》 https://www.kawabangga.com/posts/2809

举报 使用道具

回复
vimiix  新手上路  发表于 2018-4-3 14:41:55 | 显示全部楼层
延伸阅读,True,1,1.0 分别作为字典的值的时候会怎样。

举报 使用道具

回复
vimiix  新手上路  发表于 2018-4-3 15:01:22 | 显示全部楼层
https://vimiix.com/post/2017/12/28/python-mystery-dict-expression/

举报 使用道具

回复
fanhaipeng0403  新手上路  发表于 2018-4-3 15:09:59 | 显示全部楼层
印象中 None  null 这样的类型

不能用=

举报 使用道具

回复
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

快速回复 返回顶部 返回列表