声明:在人工智能技术教学期间,不少学生向我提一些python相关的问题,所以为了让同学们掌握更多扩展知识更好地理解AI技术,我让助理负责分享这套python系列教程,希望能帮到大家!由于这套python教程不是由我所写,所以不如我的AI技术教学风趣幽默,学起来比较枯燥;但它的知识点还是讲到位的了,也值得阅读!PS:看不懂本篇文章的同学请先看前面的文章,循序渐进每天学一点就不会觉得难了!

列表解析可以在for之后编写一个if分支,用来增加选择逻辑。使用了if分支的列表解析能够当成一种与内置的filter类似的工具,它们会在分支不是真的情况下跳过一些序列的元素。

这里举一个从0~4选出偶数的例子。这里的filter创建了一个小的lambda函数。另外为了对比,在后面也给出了等效的for循环实现。

>>>[x for x in range(5) if x % 2 == 0]

[0,2,4]




>>>list(filter((lambda x: x % 2 == 0),range(5)))

[0,2,4]




>>>res = []

>>>for x in range(5):

...    if x % 2 == 0:

...       res.append(x)

...

>>>res

[0,2,4]

它们都是用了求余(求除法的余数)操作符%来检测该数是否是偶数。如果一个数字除以2以后没有余数,它就一定是偶数。

>>>[x ** 2 for x in range(10) if x % 2 == 0]

[0,4,16,36,64]

上面我们收集了从0~9的偶数的平方。若在右边的if中得到的是假的话,for循环就会跳过这些数字。这个等效的map调用将需要更多的工作来完成这一部分。我们需要在map迭代中混合filter选择过程,这明显复杂得多。

>>>list( map((lambda x: x**2),filter((lambda x: x % 2 == 0),range(10))) )

[0,4,16,36,64]

实际上,你可以在一个列表解析中编写任意数量的嵌套的for循环,并且每一个都有可选的关联的if分支。通用的列表解析的结构如下所示。

[ expression for target1 in iterable1 [if condition1]

             for target2 in iterable2 [if condition2] ...

             for targetN in iterableN [if conditionN] ]

当for分句嵌套在列表解析中时,它们工作起来就像等效的嵌套的for循环语句。例如,如下代码。

>>>res = [x + y for x in [0,1,2] for y in [100,200,300]]

>>>res

[100,200,300,101,201,301,102,202,302]

与下文如此冗长的代码有相同的效果。

>>>res = []

>>>for x in [0,1,2]:

...    for y in [100,200,300]:

...        res.append(x + y)

...

>>>res

[100,200,300,101,201,301,102,202,302]

下面是个更加复杂的列表解析。

>>>[(x,y) for x in range(5) if x % 2 == 0 for y in range(5) if y % 2 == 1]

[(0,1),(0,3),(2,1),(2,3),(4,1),(4,3)]

这个表达式排列了从0~4的偶数与从0~4的奇数的组合。其中if分句过滤出了每个序列中需要进行迭代的元素。这里是一个等效的用语句编写而成的代码:

>>>res = []

>>>for x in range(5):

...     if x % 2 == 0:

...       for y in range(5):

...           if y % 2 == 1:

...               res.append((x,y))

...

>>>res

[(0,1),(0,3),(2,1),(2,3),(4,1),(4,3)]

记住!如果你对一个复杂的列表解析有什么困惑的话,你可以将列表解析的for和if分句在其中进行嵌套(将后来的分句缩进到右边),从而得到等效的语句。得到的结果要长得多,但是也更清晰。