列表推导式必须用方括号,圆括号是生成器表达式;字典推导式必须有冒号分隔键值;集合推导式去重但不保序;嵌套推导式默认扁平化,需双层方括号实现二维结构。

列表推导式写错括号就报错:[x for x in seq] 不能写成 (x for x in seq)
很多人一上来就把括号搞混,(x for x in seq) 是生成器表达式,不是列表。用在需要列表的地方(比如索引、.append()、len())会直接报 TypeError: 'generator' object is not subscriptable。
常见错误场景:想快速过滤再转成列表,却写了 result = (x for x in data if x > 0),后面又去写 result[0] —— 这时它根本不是列表。
-
[x for x in data if x > 0]→ 得到 list,支持索引、重复遍历 -
(x for x in data if x > 0)→ 得到 generator,只能遍历一次,不支持索引 - 如果数据量大且只遍历一次,用生成器更省内存;否则老老实实用方括号
字典推导式必须有 key 和 value:{k: v for k, v in items} 少一个冒号就语法错误
字典推导式的结构是 {key_expr: value_expr for item in iterable},中间的冒号是语法硬性要求,漏掉或写成逗号、等号都会触发 SyntaxError: invalid syntax。
典型翻车点:从元组列表构造字典时,误写成 {k, v for k, v in [('a',1),('b',2)]} —— 这其实是集合推导式,而且元素类型不匹配(tuple + int 混合),运行时报 TypeError: unhashable type: 'list' 或更奇怪的错误。
立即学习“Python免费学习笔记(深入)”;
- 正确写法:
{k: v for k, v in [('a',1),('b',2)]} - 如果只有键没值,想初始化为 None,得显式写:
{k: None for k in keys} - 注意 key 必须可哈希,
{list(x): 1 for x in [[1], [2]]}会崩,因为list不可哈希
集合推导式去重快,但顺序不保证:{x for x in seq} 和 list(set(seq)) 行为不同
集合推导式本质是构建 set,所以天然去重,但 Python 3.7+ 的 dict 保持插入顺序,set 不保证——哪怕输入有序,输出也可能乱序。
使用场景差异明显:做存在性检查(if x in seen_set)用集合推导最快;但想“去重并保序”,就不能直接用它,得配合 dict.fromkeys() 或手动维护。
-
{x for x in [3,1,2,1,3]}→ 可能返回{1, 2, 3},也可能{3, 1, 2}(取决于哈希扰动) - 保序去重推荐:
list(dict.fromkeys(seq)),比list(set(seq))更可靠 - 性能上,
{...}推导比set()构造略快,但差别微小,别为这点优化牺牲可读性
嵌套推导式容易读错也容易写错:[y for x in nested for y in x] 不是二维循环的直觉写法
Python 的嵌套推导式是扁平化展开的,[y for x in nested for y in x] 等价于两层 for 的缩写,不是 [[y for y in x] for x in nested]。新手常把两者混淆,结果得到一维列表而非二维。
错误示例:想把 [[1,2],[3,4]] 转成 [[10,20],[30,40]],却写了 [x*10 for row in matrix for x in row] → 结果是 [10,20,30,40],不是想要的嵌套结构。
- 要扁平化:用单层推导式,
[x for row in matrix for x in row] - 要嵌套:必须套一层推导式,
[[x*10 for x in row] for row in matrix] - 多层嵌套时,for 子句顺序和缩进 for 循环一致,但没有冒号和换行,靠空格和顺序识别逻辑层级
嵌套越深越难一眼看懂,三层以上建议拆成普通循环,别硬撑推导式。










