这两者之间有什么区别吗:

if foo is None: pass

and

if foo == None: pass

我在大多数Python代码(以及我自己编写的代码)中看到的约定是前者,但我最近遇到了使用后者的代码。None是NoneType的一个实例(也是唯一的实例,IIRC),所以它不应该有关系,对吗?有没有可能的情况?


当前回答

更多细节:

is子句实际上检查两个对象是否相同 内存位置与否。也就是它们是否都指向同一个点 内存位置和具有相同的id。 作为1的结果,is确保两个用词法表示的对象是否具有相同的属性(属性的属性…) 原始类型的实例化,如bool, int, string(有一些例外),具有相同值的NoneType将始终位于相同的内存位置。

E.g.

>>> int(1) is int(1)
True
>>> str("abcd") is str("abcd")
True
>>> bool(1) is bool(2)
True
>>> bool(0) is bool(0)
True
>>> bool(0)
False
>>> bool(1)
True

由于NoneType在python的“查找”表中只能有一个自身的实例,因此前者和后者更多的是编写代码的开发人员的编程风格(可能是为了一致性),而不是有任何微妙的逻辑原因来选择一个而不是另一个。

其他回答

is总是返回True如果它比较相同的对象实例

而==最终由__eq__()方法确定

i.e.


>>> class Foo(object):
       def __eq__(self, other):
           return True

>>> f = Foo()
>>> f == None
True
>>> f is None
False

foo为None的原因是首选的方式,因为你可能正在处理一个对象,该对象定义了自己的__eq__,并将该对象定义为None。所以,如果你需要查看它是否为None,总是使用foo is None。

a is b # returns true if they a and b are true alias
a == b # returns true if they are true alias or they have values that are deemed equivalence 


a = [1,3,4]
b = a[:] #creating copy of list
a is b # if gives false
False
a == b # gives true
True

是测试身份而不是平等对于foo语句为none, Python只是比较对象的内存地址。这意味着你在问这样一个问题:“同一个对象有两个名字吗?”

另一方面,==则测试由__eq__()方法确定的是否相等。它不关心身份。

In [102]: x, y, z = 2, 2, 2.0

In [103]: id(x), id(y), id(z)
Out[103]: (38641984, 38641984, 48420880)

In [104]: x is y
Out[104]: True

In [105]: x == y
Out[105]: True

In [106]: x is z
Out[106]: False

In [107]: x == z
Out[107]: True

None是单例操作符。所以None = None总是为真。

In [101]: None is None
Out[101]: True

更多细节:

is子句实际上检查两个对象是否相同 内存位置与否。也就是它们是否都指向同一个点 内存位置和具有相同的id。 作为1的结果,is确保两个用词法表示的对象是否具有相同的属性(属性的属性…) 原始类型的实例化,如bool, int, string(有一些例外),具有相同值的NoneType将始终位于相同的内存位置。

E.g.

>>> int(1) is int(1)
True
>>> str("abcd") is str("abcd")
True
>>> bool(1) is bool(2)
True
>>> bool(0) is bool(0)
True
>>> bool(0)
False
>>> bool(1)
True

由于NoneType在python的“查找”表中只能有一个自身的实例,因此前者和后者更多的是编写代码的开发人员的编程风格(可能是为了一致性),而不是有任何微妙的逻辑原因来选择一个而不是另一个。