python list comprehension double for

时间:2013-07-15 15:17:20

标签: python list-comprehension

vec = [[1,2,3], [4,5,6], [7,8,9]]
print [num for elem in vec for num in elem]      <----- this

>>> [1, 2, 3, 4, 5, 6, 7, 8, 9]

这是骗我的。
我理解elem是for elem in vic中列表中的列表 我不太了解numfor num in elem在开头和结尾的用法。

python如何解释这个?
它看起来的顺序是什么?

4 个答案:

答案 0 :(得分:86)

让我们分解。

一个简单的列表理解:

[x for x in collection]

如果我们将其分成几部分,这很容易理解:[A for B in C]

  • A是结果列表中的项目
  • B是集合C
  • 中的每个项目
  • C是集合本身。

通过这种方式,人们可以写:

[x.lower() for x in words]

为了将列表中的所有单词转换为小写。


当我们将其与另一个列表复杂化时,就是这样:

[x for y in collection for x in y] # [A for B in C for D in E]

在这里,会发生一些特别的事情。我们希望我们的最终列表包含A个项目,A个项目位于B个项目中,因此我们必须告诉列表理解。

  • A是结果列表中的项目
  • B是集合C
  • 中的每个项目
  • C是集合本身
  • D是集合E中的每个项目(在这种情况下,也是A
  • E是另一个集合(在本例中为B

此逻辑类似于正常for循环:

for y in collection:     #      for B in C:
    for x in y:          #          for D in E: (in this case: for A in B)
        # receive x      #              # receive A

为了扩展这个,并给出一个很好的例子+解释,想象一下有一列火车。

火车引擎(前面)总是在那里(列表理解的结果)

然后,有任意数量的火车车厢,每辆火车的形式为:for x in y

列表理解可能如下所示:

[z for b in a for c in b for d in c ... for z in y]

这就像拥有这个常规for循环:

for b in a:
    for c in b:
        for d in c:
            ...
                for z in y:
                    # have z

换句话说,在列表理解中,您只需将下一个循环添加到结尾,而不是沿着一行和缩进。

回到火车比喻:

Engine - Car - Car - Car ... Tail

尾巴是什么?尾部是列表推导中的特殊事物。你不需要一个,但如果你有一个尾巴,尾巴是一个条件,看看这个例子:

[line for line in file if not line.startswith('#')] 

只要该行没有以#标签(#开头)开头,就会跳过文件中的每一行。

使用&#34; tail&#34;的技巧火车的特点是,在您拥有最终的“发动机”的同时检查是否为真/假。或者&#39;结果&#39;从所有循环中,上面的常规for循环示例如下所示:

for line in file:
    if not line.startswith('#'):
        # have line

请注意:虽然在我对火车的比喻中只有一条尾巴&#39;在火车的尽头,条件或尾巴&#39;可以在每个&#39; car&#39;之后或循环...

例如:

>>> z = [[1,2,3,4],[5,6,7,8],[9,10,11,12]]
>>> [x for y in z if sum(y)>10 for x in y if x < 10]
[5, 6, 7, 8, 9]

在常规for循环中:

>>> for y in z:
    if sum(y)>10:
        for x in y:
            if x < 10:
                print x

5
6
7
8
9

答案 1 :(得分:8)

来自list comprehension documentation

  

提供列表推导时,它包含一个表达式,后跟至少一个for子句和零个或多个forif子句。在这种情况下,新列表的元素是通过将每个for或if子句从左到右嵌套,并且每次到达最内部块时评估表达式以生成列表元素而生成的元素。

换句话说,假装for循环是嵌套的。从左到右阅读您的列表理解可以嵌套为:

for elem in vec:
    for num in elem:
        num           # the *single expression* from the spec

其中列表推导将使用最后一个最里面的块作为结果列表的值。

答案 2 :(得分:6)

您的代码等于:

temp = []
for elem in vec:
    for num in elem:
        temp.append(num)

答案 3 :(得分:0)

您可以将列表理解视为顺序语句。这适用于任何级别的forif语句。

例如,考虑使用自己的for进行两次if循环:

vec = [[1,2,3], [4,5,6], [7,8,9]]
result = [i for e in vec if len(e)==3 for i in e if i%2==0]

列表理解如下:

result = []
for e in vec: 
    if len(e)==3:
        for i in e:
            if i%2==0:
                result.append(i)

如您所见,列表理解只是forif,没有缩进,但顺序相同。