python - 为什么 "a == x or y or z"总是评估为 True?我如何将 "a"与所有这些进行比较?

标签 python boolean boolean-expression

我正在编写一个安全系统,拒绝未经授权的用户访问。

name = input("Hello. Please enter your name: ")
if name == "Kevin" or "Jon" or "Inbar":
    print("Access granted.")
else:
    print("Access denied.")

它按预期向授权用户授予访问权限,但也允许未经授权的用户进入!

Hello. Please enter your name: Bob
Access granted.

为什么会出现这种情况?我已经明确表示仅当 name 等于 Kevin、Jon 或 Inbar 时才授予访问权限。我也尝试过相反的逻辑,如果“Kevin”或“Jon”或“Inbar”== name,但结果是相同的。

<小时/>

这个问题旨在作为这个非常常见问题的规范重复目标。还有一个热门问题How to test multiple variables for equality against a single value?具有相同的根本问题,但比较目标相反。这个问题不应该作为那个问题的重复问题而被关闭,因为这个问题是 Python 新手遇到的,他们可能很难将反向问题中的知识应用到他们的问题中。

对于in而不是==,这里有解决方案:How to test the membership of multiple values in a list

最佳答案

在许多情况下,Python 的外观和行为都类似于自然英语,但这是抽象失败的一种情况。人们可以使用上下文线索来确定“Jon”和“Inbar”是连接到动词“equals”的对象,但Python解释器更注重字面意义。

if name == "Kevin" or "Jon" or "Inbar":

逻辑上等同于:

if (name == "Kevin") or ("Jon") or ("Inbar"):

对于用户 Bob 来说,这相当于:

if (False) or ("Jon") or ("Inbar"):

or 运算符 chooses the first operand"truthy" ,即 would satisfy an if condition (或者最后一个,如果没有一个是“真实的”):

if "Jon":

由于“Jon”为真,因此 if block 将执行。这就是导致无论给出的名称如何都会打印“授予访问权限”的原因。

所有这些推理也适用于表达式if "Kevin"or "Jon"or "Inbar"== name。第一个值 "Kevin" 为 true,因此执行 if block 。

<小时/>

有两种常见方法可以正确构造此条件。

  1. 使用多个 == 运算符显式检查每个值:

    if name == "Kevin" or name == "Jon" or name == "Inbar":
    
  2. 组成有效值的集合(例如集合、列表或元组),并使用 in 运算符来测试成员资格:

    if name in {"Kevin", "Jon", "Inbar"}:
    

总的来说,在这两者中,第二个应该是首选,因为它更容易阅读并且速度更快:

>>> import timeit
>>> timeit.timeit('name == "Kevin" or name == "Jon" or name == "Inbar"',
    setup="name='Inbar'")
0.4247764749999945
>>> timeit.timeit('name in {"Kevin", "Jon", "Inbar"}', setup="name='Inbar'")
0.18493307199999265
<小时/>

对于那些可能想要证明 if a == b or c or d or e: ... 确实是这样解析的人。内置的 ast 模块提供了答案:

>>> import ast
>>> ast.parse("a == b or c or d or e", "<string>", "eval")
<ast.Expression object at 0x7f929c898220>
>>> print(ast.dump(_, indent=4))
Expression(
    body=BoolOp(
        op=Or(),
        values=[
            Compare(
                left=Name(id='a', ctx=Load()),
                ops=[
                    Eq()],
                comparators=[
                    Name(id='b', ctx=Load())]),
            Name(id='c', ctx=Load()),
            Name(id='d', ctx=Load()),
            Name(id='e', ctx=Load())]))

正如我们所看到的,它是应用于四个子表达式的 boolean 运算符or:比较a == b;以及简单表达式 cde

关于python - 为什么 "a == x or y or z"总是评估为 True?我如何将 "a"与所有这些进行比较?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57823672/

相关文章:

ruby - Ruby 中赋值时的 bool 运算

python - Slack 机器人 [Python/Flask] : Wait for Direct Messages from specified users and save the responses

C++ 编译器在使用 Bool 包装器类时报告不明确的函数调用

javascript - 如果 url 符合某些条件则重定向用户

sql - 根据 Postgres 中的另一列设置列的值?

java - 检查 Java 中的 "nested"null 属性

c++ - bool 表达式作为模板参数

跳过 Python 预提交单元测试

python - 如何在凸多边形的周长上生成随机/均匀点?

python - Unicode解码错误: 'utf-8' codec can't decode byte 0x8b in position 1