为什么 Python 中的表达式 0 < 0 == 0 返回 False?(如何解释链式比较?)

2024-12-03 08:44:00
admin
原创
179
摘要:问题描述:Queue.py我在Python 2.6标准库中的这段代码中发现了一个奇怪的结构:def full(self): """Return True if the queue is full, False otherwise (not reliable!).&...

问题描述:

Queue.py我在Python 2.6标准库中的这段代码中发现了一个奇怪的结构:

def full(self):
    """Return True if the queue is full, False otherwise
    (not reliable!)."""
    self.mutex.acquire()
    n = 0 < self.maxsize == self._qsize()
    self.mutex.release()
    return n

显然,如果maxsize为 0,队列就永远不会满。但这是如何工作的呢?为什么0 < 0 == 0求值为 False?无论先应用哪个操作,结果似乎都应该是 True,而且我确实在添加括号时得到了这个结果:

>>> 0 < 0 == 0
False
>>> (0 < 0) == 0
True
>>> 0 < (0 == 0)
True

另请参阅“x < y < z”是否比“x < y and y < z”更快?以了解有关如何实现该功能的详细信息。


解决方案 1:

Python 对关系运算符序列有特殊情况处理,使范围比较更容易表达。能够说0 < x <= 5比说要好得多(0 < x) and (x <= 5)

这些被称为链式比较。

对于您谈到的其他情况,括号会强制先应用一个关系运算符,然后再应用另一个关系运算符,因此它们不再是链式比较。由于TrueFalse的值都是整数,因此您可以从带括号的版本中获得答案。

解决方案 2:

因为

(0 < 0) and (0 == 0)

False。您可以将比较运算符链接在一起,它们会自动扩展到成对比较中。


编辑——关于 Python 中 True 和 False 的说明

在 Python 中True,和False只是 的实例bool,而 是 的子类int。换句话说,True实际上只是 1。

这样做的目的是,你可以像使用整数一样使用布尔比较的结果。这会导致一些令人困惑的事情,例如

>>> (1==1)+(1==1)
2
>>> (2<1)<1
True

但只有当你用括号括住比较运算符以便首先对其进行求值时,才会发生这种情况。否则 Python 将扩展比较运算符。

解决方案 3:

您遇到的奇怪行为来自 Python 链接条件的能力。由于它发现 0 不小于 0,因此它决定整个表达式的计算结果为 false。一旦将其分解为单独的条件,就会改变功能。它最初本质上是测试a < b && b == c您原始的 语句a < b == c

另一个例子:

>>> 1 < 5 < 3
False

>>> (1 < 5) < 3
True

解决方案 4:

>>> 0 < 0 == 0
False

这是一个链式比较。如果每个成对比较依次为真,则返回真。它相当于(0 < 0) and (0 == 0)

>>> (0) < (0 == 0)
True

这相当于0 < True计算结果为 True。

>>> (0 < 0) == 0
True

这相当于False == 0计算结果为 True。

>>> 0 < (0 == 0)
True

0 < True与上述等同,计算结果为 True。

解决方案 5:

通过查看反汇编(字节代码),很明显为什么0 < 0 == 0False

以下是对这个表达式的分析:

>>>import dis

>>>def f():
...    0 < 0 == 0

>>>dis.dis(f)
  2      0 LOAD_CONST               1 (0)
         3 LOAD_CONST               1 (0)
         6 DUP_TOP
         7 ROT_THREE
         8 COMPARE_OP               0 (<)
        11 JUMP_IF_FALSE_OR_POP    23
        14 LOAD_CONST               1 (0)
        17 COMPARE_OP               2 (==)
        20 JUMP_FORWARD             2 (to 25)
   >>   23 ROT_TWO
        24 POP_TOP
   >>   25 POP_TOP
        26 LOAD_CONST               0 (None)
        29 RETURN_VALUE

注意第 0-8 行:这些行检查是否0 < 0显然返回False到 python 堆栈。

现在注意第 11 行:JUMP_IF_FALSE_OR_POP 23
这意味着如果0 < 0返回False则跳转到第 23 行。

现在,0 < 0False,因此进行跳转,这会使堆栈留下一个False,它是整个表达式的返回值0 < 0 == 0,即使该== 0部分甚至没有被检查。

因此,总而言之,答案就像这个问题的其他答案中所说的那样。
0 < 0 == 0具有特殊含义。编译器将其评估为两个项:0 < 00 == 0。与and它们之间的任何复杂布尔表达式一样,如果第一个失败,则第二个甚至不会被检查。

希望这能对大家有所启发,我也真心希望我用来分析这种意外行为的方法能够鼓励其他人在未来尝试同样的做法。

解决方案 6:

正如其他人提到的那样,x comparison_operator y comparison_operator z这是一种语法糖(x comparison_operator y) and (y comparison_operator z),其好处是 y 只需要评估一次。

因此,您的表达式0 < 0 == 0实际上是(0 < 0) and (0 == 0),其计算结果False and True恰好是False

解决方案 7:

也许文档中的这段摘录可以提供帮助:

这些是所谓的“丰富比较”方法,在以下情况下优先调用比较运算符__cmp__()。运算符号与方法名的对应关系如下:x<ycalls
x.__lt__(y)x<=ycalls x.__le__(y)
x==ycalls x.__eq__(y)x!=ycalls 、calls
、calls
x<>y`x.__ne__(y)x>yx.__gt__(y)x>=yx.__ge__(y)`

如果富比较方法NotImplemented未针对给定的一对参数实现操作,则可能会返回单例。按照惯例,False比较True成功时会返回和。但是,这些方法可以返回任何值,因此如果在布尔上下文中使用比较运算符(例如,在 if 语句的条件中),Python 将调用bool()该值来确定结果是真还是假。

比较运算符之间没有隐含关系。 的真值x==y并不意味着 的x!=y
假值。因此,在定义 时
__eq__(),还应定义__ne__()以使运算符的行为符合预期。请参阅 段落,了解__hash__()有关创建支持自定义比较操作并可用作字典键的可哈希对象的一些重要说明。

这些方法没有交换参数的版本(当左参数不支持操作但右参数支持时使用);相反,__lt__()__gt__()
是彼此的反射,__le__()
__ge__()是彼此的反射,和__eq__()__ne__()
它们自己的反射。

丰富的比较方法的论点永远不会被强迫。

这些都是比较,但由于您正在进行链接比较,因此您应该知道:

比较可以任意链接,例如,x < y <= z相当于x < y and y <= z,除了 y 仅被评估一次(但在这两种情况下,当发现 x < y 为假时,z 根本不被评估)。

形式上,如果 a、b、c、...、y、z 是表达式,而 op1、op2、...、opN 是比较运算符,则 a op1 b op2 c ... y opN z 等同于 a op1 b and b op2 c and ... y opN z,不同之处在于每个表达式最多被求值一次。

解决方案 8:

这就是它的全部辉煌。

>>> class showme(object):
...   def __init__(self, name, value):
...     self.name, self.value = name, value
...   def __repr__(self):
...     return "<showme %s:%s>" % (self.name, self.value)
...   def __cmp__(self, other):
...     print "cmp(%r, %r)" % (self, other)
...     if type(other) == showme:
...       return cmp(self.value, other.value)
...     else:
...       return cmp(self.value, other)
... 
>>> showme(1,0) < showme(2,0) == showme(3,0)
cmp(<showme 1:0>, <showme 2:0>)
False
>>> (showme(1,0) < showme(2,0)) == showme(3,0)
cmp(<showme 1:0>, <showme 2:0>)
cmp(<showme 3:0>, False)
True
>>> showme(1,0) < (showme(2,0) == showme(3,0))
cmp(<showme 2:0>, <showme 3:0>)
cmp(<showme 1:0>, True)
True
>>> 

解决方案 9:

我认为 Python 在魔法之间做了一些奇怪的事情。与1 < 2 < 32 介于 1 和 3 之间相同。

在这种情况下,我认为它正在执行 [中间 0] 大于 [左 0] 且等于 [右 0]。中间 0 不大于左 0,因此其计算结果为 false。

相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   1565  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1354  
  信创国产芯片作为信息技术创新的核心领域,对于推动国家自主可控生态建设具有至关重要的意义。在全球科技竞争日益激烈的背景下,实现信息技术的自主可控,摆脱对国外技术的依赖,已成为保障国家信息安全和产业可持续发展的关键。国产芯片作为信创产业的基石,其发展水平直接影响着整个信创生态的构建与完善。通过不断提升国产芯片的技术实力、产...
国产信创系统   21  
  信创生态建设旨在实现信息技术领域的自主创新和安全可控,涵盖了从硬件到软件的全产业链。随着数字化转型的加速,信创生态建设的重要性日益凸显,它不仅关乎国家的信息安全,更是推动产业升级和经济高质量发展的关键力量。然而,在推进信创生态建设的过程中,面临着诸多复杂且严峻的挑战,需要深入剖析并寻找切实可行的解决方案。技术创新难题技...
信创操作系统   27  
  信创产业作为国家信息技术创新发展的重要领域,对于保障国家信息安全、推动产业升级具有关键意义。而国产芯片作为信创产业的核心基石,其研发进展备受关注。在信创国产芯片的研发征程中,面临着诸多复杂且艰巨的难点,这些难点犹如一道道关卡,阻碍着国产芯片的快速发展。然而,科研人员和相关企业并未退缩,积极探索并提出了一系列切实可行的解...
国产化替代产品目录   28  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用