高级语法

除了像上面介绍的 [x ** 2 for x in L] 这种基本语法之外,列表推导式还有一些高级的扩展。

4.1. 带有if语句

我们可以在 for 语句后面跟上一个 if 判断语句,用于过滤掉那些不满足条件的结果项。

例如,我想去除列表中所有的偶数项,保留奇数项,可以这么写:

  1. >>> L = [1, 2, 3, 4, 5, 6]
  2. >>> L = [x for x in L if x % 2 != 0]
  3. >>> L
  4. [1, 3, 5]

4.2. 带有for嵌套

在复杂一点的列表推导式中,可以嵌套有多个 for 语句。按照从左至右的顺序,分别是外层循环到内层循环。

例如:

  1. >>> [x + y for x in 'ab' for y in 'jk']
  2. ['aj', 'ak', 'bj', 'bk']

4.3. 既有if语句又有for嵌套

列表推导式可以带任意数量的嵌套 for 循环,并且每一个 for 循环后面都有可选的 if 语句。

通用语法:

  1. [ expression for x in X [if condition]
  2. for y in Y [if condition]
  3. ...
  4. for n in N [if condition] ]

例如,下面的代码输出了0~4之间的偶数和奇数的组合。

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

等价于下面的一般 for 循环:

  1. >>> L = []
  2. >>> for x in range(5):
  3. ... if x % 2 == 0:
  4. ... for y in range(5):
  5. ... if y % 2 == 1:
  6. ... L.append((x, y))
  7. >>> L
  8. [(0, 1), (0, 3), (2, 1), (2, 3), (4, 1), (4, 3)]

4.4. 列表推导式生成矩阵

生成矩阵的方式有多种,例如手动赋值、一般for循环,还有就是列表推导式。如果我们要用列表推导式生成下面的矩阵,可以怎么写?

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

一种方法是:

  1. >>> M = [[x, x+1, x+2] for x in [1, 4, 7]]
  2. >>> M
  3. [[1, 2, 3], [4, 5, 6], [7, 8, 9]]

矩阵的列数少时可以使用这种方法。

如果矩阵的列数较多,我们可以使用另外一种方式:在循环变量的表达式中使用列表推导式。

具体代码如下:

  1. >>> M = [[y for y in range(x, x+3)] for x in [1, 4, 7]]
  2. >>> M
  3. [[1, 2, 3], [4, 5, 6], [7, 8, 9]]

与之前带 for 嵌套的语法不同,这个例子中,实际使用的是最基本的 [expression for x in L] 语法,只有一个 for 语句。

复杂的地方在于前面的变量表达式 expression 不再是简单的变量运算,而是一个列表推导式,在这个例子中就是 [y for y in range(x, x+3)]
内层的列表推导式返回一个行向量,而这些行向量经由外层的列表推导式,最终形成一个二维列表,也就是我们想要的矩阵。

当然,在实际的应用中不能单纯追求代码的简洁,还要考虑到代码的可读性和维护成本。
如果代码变得过于复杂,不易于理解,我们宁可多写几行代码来增加它的可读性。

5. 生成器表达式

生成器表达式与列表推导式的语法相同,区别在于生成器表达式的外面使用圆括号,而列表推导式使用方括号。

有关生成器的介绍,请参考这篇文章:《Python高级编程之初识生成器》

6. 集合推导式和字典推导式

注意:集合推导式和字典推导式只有在Python2.7以及之后的版本中才有,Python2.7之前的版本不支持这两种推导式。

集合推导式的语法与列表推导式相同,只需要把外面的方括号改成花括号即可。

例如,我们可以通过以下方式来生成一个集合:

  1. >>> {x ** 2 for x in [1, 2, 2]}
  2. {1, 4}

字典推导式的外面也是使用花括号,不过花括号的内部需要包含键值两部分。

在值不重复的情况下,我们可以通过字典推导式快速交换键值对:

  1. >>> D = {'a':1, 'b':2, 'c':3}
  2. >>> D = {value: key for key, value in D.items()}
  3. >>> D
  4. {1: 'a', 2: 'b', 3: 'c'}

from:http://www.codebelief.com/article/2017/02/python-advanced-programming-list-comprehensions/


  1. lix = [];
  2. for x in range(1, 101):
  3. lix.push(x ** 2)
  4. 执行结果:lix = [1,4,9,16,25.....]

在列表构建器的表达式中,可以添加简单的条件处理

  1. lix = [x * x for x in range(1, 101) if x % 2 == 0]
  2. 执行结果:lix = [4,16,36.....]

也可以在循环过程中,来使用多层循环嵌套,实现更加复杂的效果

  1. lix = [x + y for x in "abc" for y in "xyz"]
  2. 执行结果:['ax', 'ay', 'az', 'bx', 'by', 'bz', 'cx', 'cy', 'cz']

列表动态构建器

但是我们通过前面的学习已经知道,这些数据都是加载到内存中的,如果列表中的数据量比较大的情况下,内存消耗是比较严重的
在某些情况下,我们只需要使用列表中的一部分数据,后面的数据并不是特别关心,如:通过列表来记录一个符合某种规则的序列,每次我们只是关心下一个数据,并不关心后面的N条数据,应该怎么做呢?比如我们需要一个奇数列表

  1. # 常规构建器的做法
  2. lix = [2*x + 1 for x in range(1, 101)]
  3. # 执行结果:[1,3,5,7,9,11,13,15,17.....]
  4. # * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
  5. # 常规构建器可以直接构建生成
  6. # 但是存在问题,如果一次构建的数据量太大,会严重占用内存
  7. # 我们在使用该列表的时候,有可能只是使用前10项
  8. # * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
  9. # 使用列表动态构建器
  10. lix = (2 * x - 1 for x in range(1, 101))
  11. # 执行结果:print (lix) --> <generator object <genexpr> at 0x7f232e462048>
  12. next(lix)
  13. # 执行结果:1
  14. next(lix)
  15. # 执行结果:3
  16. next(lix)
  17. # 执行结果:5
  18. next(lix)
  19. # 执行结果:7
  20. # * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
  21. # 列表动态构建器
  22. # 和构建器基本没有区别,创建的时候列表中是没有数据的
  23. # 必须通过next()函数来获取列表中的下一条数据
  24. # * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *

from:https://www.jianshu.com/p/fa3fda487f15

 
 

[]改成(),就创建了一个generator:

  1. >>> L = [x * x for x in range(10)]
  2. >>> L
  3. [0, 1, 4, 9, 16, 25, 36, 49, 64, 81]
  4. >>> g = (x * x for x in range(10))
  5. >>> g
  6. <generator object <genexpr> at 0x1022ef630>

L和g的区别仅在于最外层的[]()L是一个list,而g是一个generator。

如果要一个一个打印出来,可以通过next()函数获得generator的下一个返回值:

  1. >>> next(g)
  2. 0
  3. >>> next(g)
  4. 1
  5. >>> next(g)
  6. 4
  7. >>> next(g)
  8. 9
  9. >>> next(g)

next(g)实在是太变态了,正确的方法是使用for循环,因为generator也是可迭代对象:

  1. >>> g = (x * x for x in range(10))
  2. >>> for n in g:
  3. ... print(n)
  4. ...
  5. 0
  6. 1
  7. 4
  8. 9
    要把fib函数变成generator,只需要把print(b)改为yield b就可以了:
  1. def fib(max):
  2. n, a, b = 0, 0, 1
  3. while n < max:
  4. yield b
  5. a, b = b, a + b
  6. n = n + 1
  7. return 'done'

yield关键字,那么这个函数就不再是一个普通函数,而是一个generator:

  1. >>> f = fib(6)
  2. >>> f
  3. <generator object fib at 0x104feaaa0>

return语句或者最后一行函数语句就返回。而变成generator的函数,在每次调用next()的时候执行,遇到yield语句返回,再次执行时从上次返回的yield语句处继续执行。

  1.  

python 循环高级用法 [expression for x in X [if condition] for y in Y [if condition] ... for n in N [if condition] ]按照从左至右的顺序,分别是外层循环到内层循环的更多相关文章

  1. python requests 高级用法

    高级用法 本篇文档涵盖了 Requests 的一些高级特性. 会话对象 会话对象让你能够跨请求保持某些参数.它也会在同一个 Session 实例发出的所有请求之间保持 cookie, 期间使用 url ...

  2. Python pip高级用法

    1.pip 高级用法为了便于用户安装和管理第三方库和软件,越来越多的编程语言拥有自己的包管理工 具,如 nodejs 的 npm, ruby 的 gem. Python 也不例外,现在 Python ...

  3. Volocity循环高级用法

    #foreach($announcementDo in $announcementList) #set($listSize=$!announcementList.size() - 1) #if(($v ...

  4. (转)python requests 高级用法 -- 包括SSL 证书错误的解决方案

    我在使用requests访问某个https网站时出现错误  error::SSL routines:SSL3_GET_SERVER_CERTIFICATE:certificate verify fai ...

  5. python requests 高级用法 -- 包括SSL 证书错误的解决方案

    Session Objects会话对象 Session对象在请求时允许你坚持一定的参数.此外,还坚持由Session实例的所有请求的cookie. 让我们坚持在请求时使用 s = requests.S ...

  6. Python sqlalchemy 高级用法

    一. 关联查询 sys_user_list = SysPermission.query.join(OrgRolePermission, OrgRolePermission.sys_permission ...

  7. python核心高级学习总结7---------正则表达式

    正则表达式在爬虫项目中应用很广泛,主要方面就是在字符串处理方面,经常会涉及到字符串格式的校验,用起来经常要查看文档才能完成,所以抽了个时间将正则的内容复习了一下. Start re---导入re模块使 ...

  8. 四年级--python函数基础用法

    一.函数的定义,调用和返回值 1.1 语法 def 函数(参数一,参数二...): ''' 文档注释 ''' 代码逻辑一 代码逻辑二 .... return 返回值 1.2 定义函数的三种形式 说明: ...

  9. Python内建函数enumerate()用法及在for循环应用

    Python 内建函数enumerate() 由于这个单纯很长,不容易记住,用法还是比较广泛的,下面讲述Python内建函数enumerate()用法. 1,实例 enumerate(sequence ...

随机推荐

  1. win10系统杀毒功能

    最近很多客户的电脑都是win10的,部署web服务时用到的破解软件(exe文件),经常被当成病毒杀死,今天发现可以在杀毒记录里找到刚刚被杀掉的文件,然后点击操作,点击允许,再运行时就可以畅通无阻了

  2. returnValue of Chrome

    说实话,我一看到这个returnValue就有点反感,感觉这个就是IE式的老套的用法,因为项目中有用到就了解了下,以下主要是一些我的理解和发现吧. PS:returnValue是window的属性,s ...

  3. 攻破javascript面试的完美指南【译】

    攻破javascript面试的完美指南(开发者视角) 0. 前言 本文适合有一定js基础的前端开发人员阅读.原文是我google时无意发现的, 被一些知识点清晰的解析所打动, 决定翻译并记录下来.这个 ...

  4. 移动端开发需要加的meta

    移动端开发需要加的meta和常用的css3媒体查询样式,移动开发中头部要加的一些常用meta. <meta name="viewport" content="ini ...

  5. C语言输入一行整数(OJ输入格式)

    就是说输入一行用空格隔开的函数,可是它没说用回车符结束,所以一定要用EOF了 第一种方法: ; char ch; do { scanf("%ld",&a[++t]); } ...

  6. layui 动态表格之合并单元格

    需求: 下面用excel表格大概模拟下需求,左边是原来的,要改成右边这样的: ①第一步:再生成表格后调用此方法,以合并重复的单元格 done : function(res, curr, count) ...

  7. Mac安装Qt出现错误Could not resolve SDK Path for 'macosx'

    Qt 5.8 + Mac 10.14  qdevice.pri文件里没有网上说的那行应该改的代码,自己写上这句话也没有解决问题 最终解决方案: 在命令行输入:sudo xcode-select -s ...

  8. gnulpot

    gnulpot Table of Contents 1. Label position 2. coordinates 3. Symbols 4. key 4.1. key position 4.2. ...

  9. 基于 Ubuntu 搭建 FTP 文件服务

    搭建 FTP 文件服务 安装并启动 FTP 服务 任务时间:5min ~ 10min 安装 VSFTPD 使用 apt-get 安装 vsftpd: sudo apt-get install vsft ...

  10. 3.2.2.5 BRE运算符优先级

        在数学表达式里,正则表达式的运算符具有某种已定义的优先级,指的是某个运算符(优先级较高)将比其他运算符先被处理.       BRE运算符优先级,由高至低   运算符 表示含义 [..] [= ...