什么是列表推导?
列表推导是一个将一个列表(实际上是任意可迭代对象)转换成另一个列表的工具。在转换时,每个元素都可以按照某个条件被包含在新的列表中,并根据需要做出一些变换。
如果你熟悉函数式编程,你可以把列表推导想成是一个filter
后面跟了一个map的语法糖
1
2
|
>>> doubled_odds = map(lambda n: n * 2, filter(lambda n: n % 2 == 1, numbers))
>>> doubled_odds = [n * 2 for n in numbers if n % 2 == 1]
|
如果你不熟悉函数式编程,别担心,我会用for
循环来解释。
从循环到列表推导
每个列表推导都可以重写成for
循环的形式,但并不是每一个for
循环都可以重写成列表推导。
要理解什么时候该使用列表推导,关键在于不断练习辨别哪些问题看起来像是列表推导。
如果你能把你的代码重写成这个样子的for
循环,那你就能把它重写成列表推导:
1
2
3
4
|
new_things = []
for ITEM in old_things:
if condition_based_on(ITEM):
new_things.append("something with " + ITEM)
|
你可以把上面的for
循环重写成这个样子的列表推导:
1
|
new_things = ["something with " + ITEM for ITEM in old_things if condition_based_on(ITEM)]
|
列表推导:动态图™
看起来不错,不过具体要怎么做呢?
我们用复制粘贴来把一个for
循环变成列表推导。
下面是复制粘贴的顺序:
- 复制给新列表赋值的语句(第三行)
- 把我们
append
到新列表的表达式复制过来(第六行) - 复制
for
循环的那一行,除去末尾的:
(第四行) - 复制
if
语句,除去末尾的:
(第五行)
现在我们就把下面这个for 循环:
1
2
3
4
5
6
|
numbers = [1, 2, 3, 4, 5]
doubled_odds = []
for n in numbers:
if n % 2 == 1:
doubled_odds.append(n * 2)
|
变成了这个样子:
1
2
3
|
numbers = [1, 2, 3, 4, 5]
doubled_odds = [n * 2 for n in numbers if n % 2 == 1]
|
列表推导:现在加上颜色
我们来给代码加上高亮。
doubled_odds = []
for n in numbers:
if n % 2 == 1:
doubled_odds.append(n * 2)
doubled_odds = [n * 2 for n in numbers if n % 2 == 1]
- 复制给新列表赋值的语句
- 把我们
append
到新列表的表达式复制过来 - 复制
for
循环的那一行,除去末尾的:
- 复制
if
语句,同样去掉:
无条件列表推导
但如果是没有条件语句(就是末尾的那个if SOMETHING
)的情形呢?这些循环添加元素的for
循环比我们刚刚讲过的那种循环并根据条件添加元素的要更简单。
没有if
语句的for
循环:
1
2
3
|
doubled_numbers = []
for n in numbers:
doubled_numbers.append(n * 2)
|
同样的代码写成列表推导:
1
|
doubled_numbers = [n * 2 for n in numbers]
|
下面是变换的动态图:
我们可以按照下列步骤从for
循环里复制粘贴:
- 复制给新列表赋值的语句(第三行)
- 把我们
append
到新列表的表达式复制过来(第五行) - 复制
for
循环的那一行,除去末尾的:
(第四行)
嵌套循环
如果是带有嵌套循环的列表推导呢?……
这是把一个矩阵平铺成向量的for
循环:
1
2
3
4
|
flattened = []
for row in matrix:
for n in row:
flattened.append(n)
|
这是相应的列表推导:
1
|
flattened = [n for row in matrix for n in row]
|
列表推导里的嵌套循环读起来并不像是英语散文那样通俗易懂。
注意:我在脑袋里想把这个列表推导写成这个样子:
1
|
flattened = [n for n in row for row in matrix]
|
但是这是不对的!这里我错把两个循环颠倒了,上面的那个才是正确的。
在处理列表推导里的嵌套for
循环时,要记住:for
语句的顺序和原来的循环中for
语句的顺序是一样的。
其他推导式
这个原则同样适用于集合推导和字典推导。
下面的代码创建了一个由一个序列中所有单词的首字母组成的集合:
1
2
3
|
first_letters = set()
for w in words:
first_letters.add(w[0])
|
写成集合推导:
1
|
first_letters = {w[0] for w in words}
|
创建一个新的字典把原来字典的值和键值交换的代码:
1
2
3
|
flipped = {}
for key, value in original.items():
flipped[value] = key
|
写成字典推导:
1
|
flipped = {value: key for key, value in original.items()}
|
可读性很重要
你有没有发现上面的这些列表推导可读性比较差?当那些比较长的列表推导写在一行里的时候,我经常觉得它们很难读懂。
记住,Python 允许在括号之间进行换行。
列表推导
换行前
1
|
doubled_odds = [n * 2 for n in numbers if n % 2 == 1]
|
换行后
1
2
3
4
5
|
doubled_odds = [
n * 2
for n in numbers
if n % 2 == 1
]
|
列表推导中的嵌套循环
换行前
1
|
flattened = [n for row in matrix for n in row]
|
换行后
1
2
3
4
5
|
flattened = [
n
for row in matrix
for n in row
]
|
字典推导
换行前
1
|
flipped = {value: key for key, value in original.items()}
|
换行后
1
2
3
4
|
flipped = {
value: key
for key, value in original.items()
}
|
注意,我们不是随便换行的:在写列表推导的时候我们复制粘贴了很多条语句,我们是在这些语句之间换行的。在用各种颜色标注的版本中,我们是在颜色发生变化时换行的。