列表推导式中三元表达式必须放在for前,如[x2 if x%2==0 else -x for x in [1,2,3,4]];若仅需过滤则用[x2 for x in lst if x>0],二者语法和语义均不同。
很多人写成 [x if x > 0 for x in lst],结果报错 SyntaxError: invalid syntax。这是因为三元表达式(value_if_true if condition else value_if_false)在列表推导式里只能作为“元素生成逻辑”,必须紧挨着左方的表达式,不能跟在 for 后面。
想对每个元素做“满足条件取 A,否则取 B”的映射,写法是:
[x * 2 if x %2 == 0 else -x for x in [1, 2, 3, 4]]
结果是 [ -1, 4, -3, 8 ]。注意几点:
if 和 else 是配套出现的,缺一不可x * 2 if x % 2 == 0 else -x 是一个整体,算作推导式的“输出项”if 不写 else——那属于过滤(filtering),语法和用途完全不同这是最容易出错的地方。下面两种写法语义完全不同:
[x * 2 for x in lst if x > 0] → 只处理正数,负数直接跳过(长度可能变短)[x * 2 if x > 0 else 0 for x in lst] → 所有元素都保留,正数翻倍、非正数填 0(长度不变)如果混写成 [x * 2 if x > 0 else 0 for x in lst if x != 0],就既是映射又是过滤,逻辑变复杂,可读性下降,调试时容易漏掉某类边界值。
当条件分支超过两层(比如 if A: X elif B: Y else Z),或涉及多个变量、函数调用、副作用操作时,硬塞进一行推导式会严重降低可维护性。例如:
[f(x) if x > 10 else g(x) if x > 0 else h(x) for x in data]
这种写法虽然合法,但已经失去推导式的简洁优势。不如写成:
result = []
for x in data:
if x > 10:
result.append(f(x))
elif x > 0:
result.append(g(x))
else:
result.append(h(x))
真正需要关注的不是“能不能写成一行”,而是“别人(包括三天后的你自己)能不能一眼看懂执行路径”。