Python中and和or返回实际操作数而非布尔值:a and b在a真时返b、否则返a;a or b在a真时返a、否则返b;短路求值从左到右,一旦结果确定即停止后续求值。

Python中and和or不是返回True/False
很多人误以为and和or只做逻辑判断并返回布尔值,其实它们返回的是**参与运算的实际操作数**。这是理解短路行为的前提。
规则很简单:
-
a and b:如果a为真值(truthy),返回b;否则返回a -
a or b:如果a为真值,返回a;否则返回b
所谓“真值”,指非空容器、非零数字、非None、非False等——不是仅看类型是否为bool。
print(0 and "hello") # 输出:0
print([] or [1, 2]) # 输出:[1, 2]
print("a" and [3]) # 输出:[3]
print(None or "fallback") # 输出:"fallback"短路发生时机:从左到右逐个求值
短路不是“跳过右边”,而是**一旦能确定整个表达式结果,就停止求值后续操作数**。这直接影响副作用(如函数调用、打印、修改变量)是否执行。
立即学习“Python免费学习笔记(深入)”;
典型陷阱场景:
-
func1() and func2():若func1()返回False、0或[],func2()根本不会被调用 -
func1() or func2():若func1()返回真值,func2()不执行 - 混合使用时,优先级
not > and > or,但求值顺序始终是左→右
def log_and_return(x):
print(f"called with {x}")
return x
下面只打印一次
result = log_and_return(0) and log_and_return("never reached")
输出:called with 0
result == 0
常见误用:把and/or当三元运算符写法
有人用a and b or c模拟b if a else c,但这在b为假值时会出错。
比如:
x = [] and "yes" or "no" # 期望"no",实际输出"no" ✅ y = "ok" and [] or "no" # 期望[],但输出"no" ❌(因为[]是falsy,触发or分支)
正确替代方案只有:
- 显式用条件表达式:
"yes" if x else "no" - 避免依赖
and/or的返回值做业务逻辑分支
尤其注意字符串""、数字0、空列表[]、None都可能意外触发or分支。
嵌套与括号:改变求值分组,不改变左结合性
Python中and和or都是左结合,即a and b and c等价于(a and b) and c,不是a and (b and c)。但括号可强制分组,影响短路位置。
例如:
a = False
b = print("B")
c = print("C")
无括号:a and b or c → (a and b) or c → False or c → 执行c
有括号:a and (b or c) → a and (...) → 因a为False,b和c都不执行
关键点:括号改的是**语法分组**,不是求值顺序方向。所有表达式仍严格从左到右扫描,只是“哪几个操作数属于同一层运算”由括号决定。
真正容易忽略的是:短路只发生在and或or运算符的**右侧操作数**上,左侧永远先求值——哪怕左侧是另一个and表达式。










