有时候,一个编程设计模式使用得十分普遍,甚至会逐步形成自己独特的语法。Python编程语言中的列表解析式(list comprehension)就是这类语法糖(syntactic sugar)的绝佳代表。

Python中的 列表解析式 是个伟大的发明,但是要掌握好这个语法则有些难,因为它们并是用来解决全新的问题:只是为解决已有问题提供了新的语法。

接下来,我们一起来学习什么是列表解析式,以及如何掌握使用这种语法的时机。

什么是列表解析式?

列表解析式是将一个列表(实际上适用于任何 可迭代对象(iterable) )转换成另一个列表的工具。在转换过程中,可以指定元素必须符合一定的条件,才能添加至新的列表中,这样每个元素都可以按需要进行转换。

如果你熟悉函数式编程(functional programming),你可以把列表解析式看作为结合了 filter 函数与 map 函数功能的语法糖:

>>> 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 循环都能重写为列表解析式。

掌握列表解析式使用时机的关键,在于不断练习识别那些看上去像列表解析式的问题(practice identifying problems that smell like list comprehensions)。

如果你能将自己的代码改写成类似下面这个 for 循环的形式,那么你也就可以将其改写为列表解析式:

new_things = []
for ITEM in old_things:if condition_based_on(ITEM):new_things.append("something with " + ITEM)

你可以将上面的 for 循环改写成这样的列表解析式:

new_things = ["something with " + ITEM for ITEM in old_things if condition_based_on(ITEM)]

列表解析式:可视化解读

可视化解读听上去是个不错的注意,但是我们怎么才能做到这点呢?

嘿嘿,只需要从 for 循环中复制粘贴,稍微调整一下就变成了列表解析式啦。

下面是我们复制粘贴的顺序:

  • 将变量赋值操作复制到新建的空列表中(第三行)
  • 将 append() 方法中的表达式参数复制到新列表中(第六行)
  • 复制 for 循环语句,不包括最后的 : (第四行)
  • 复制 if 条件控制语句,同样不包括最后的 : (第五行)

这样,我们将从下面这段代码:

numbers = [1, 2, 3, 4, 5]doubled_odds = []
for n in numbers:if n % 2 == 1:doubled_odds.append(n * 2)

转换成了这两行代码:

numbers = [1, 2, 3, 4, 5]doubled_odds = [n * 2 for n in numbers if n % 2 == 1]

无条件子句的列表解析式

如果是那些没有条件子句(即 if SOMETHING 部分)的代码呢,又该怎样复制粘贴?这些形式的代码甚至比有条件子句的代码更好实现。

一个没有 if 语句的 for 循环:

doubled_numbers = []
for n in numbers:doubled_numbers.append(n * 2)

上面这段代码页可以改写为一个列表解析式:

doubled_numbers = [n * 2 for n in numbers]

下面是转换过程的详细演示:

我们可以从上面那个简单的 for 循环中,安装这样的顺序复制粘贴:

  • 将变量赋值操作复制到新建的空列表中(第三行)
  • 将 append() 方法中的表达式参数复制到新列表中(第五行)
  • 复制 for 循环语句,不包括最后的 : (第四行)

嵌套循环

那么嵌套循环(nested loop)又该怎样改写为列表解析式呢?

下面是一个拉平(flatten)矩阵(以列表为元素的列表)的 for 循环:

flattened = []
for row in matrix:for n in row:flattened.append(n)

下面这个列表解析式实现了相同的功能:

flattened = [n for row in matrix for n in row]

列表解析式中的嵌套循环读起来就有点绕口了。

注意:我本能地会想把这个列表解析式写成这样:

flattened = [n for n in row for row in matrix]

但是这行代码是错误的。这里我不小心颠倒了两个 for 循环的顺序。正确的代码是之前那个。

如果要在列表解析式中处理嵌套循环,请记住 for 循环子句的顺序与我们原来 for 循环的顺序是一致的 。

同样地原则也适用集合解析式(set comprehension)和字典解析式(dictionary comprehension)。

其他解析式

下面的代码提取单词序列中每个单词的首字母,创建了一个集合(set):

first_letters = set()
for w in words:first_letters.add(w[0])

同样的代码可以改写为集合解析式:

first_letters = {w[0] for w in words}

下面的代码将原有字典的键和值互换,从而创建了一个新的字典:

flipped = {}
for key, value in original.items():flipped[value] = key

同样的代码可以改写为字典解析式:

flipped = {value: key for key, value in original.items()}

还要注意可读性

你有没有发现上面的列表解析式读起来很困难?我经常发现,如果较长的列表解析式写成一行代码,那么阅读起来就非常困难。

不过,还好Python支持在括号和花括号之间断行。

列表解析式 List comprehension

断行前:

doubled_odds = [n * 2 for n in numbers if n % 2 == 1]

断行后:

doubled_odds = [n * 2for n in numbersif n % 2 == 1
]

带嵌套循环的列表解析式

断行前:

flattened = [n for n in row for row in matrix]

断行后:

flattened = [nfor row in matrixfor n in row
]

字典解析式

断行前:

flipped = {value: key for key, value in original.items()}

断行后:

flipped = {value: keyfor key, value in original.items()
}

请注意,我们并不是随意进行断行:我们是在每一行复制过来的代码之后断行的。

总结

纠结于写不出列表解析式吗?不要担心。先写一个 for 循环,能后按照本文说的顺序复制粘贴,就可以写出解析式了。

任何类似下面代码形式的 for 循环:

new_things = []
for ITEM in old_things:if condition_based_on(ITEM):new_things.append("something with " + ITEM)

都可以被改写为下面这种列表解析式:

new_things = ["something with " + ITEM for ITEM in old_things if condition_based_on(ITEM)]

Python列表解析式相关推荐

  1. python列表解析式_Python基础入门-列表解析式

    今天我们使用Python中的列表解析式来实现一些简单功能.好了关于列表解析式是什么?我的理解是它可以根据已有列表,高效创建新列表的方式.列表解析是Python迭代机制的一种应用,它常用于实现创建新的列 ...

  2. Python列表解析式总结

    前言 目录 前言 什么是列表解析式? 从循环到解析式 列表解析式:可视化解读 无条件子句的列表解析式 嵌套循环 其他解析式 还要注意可读性 列表解析式 List comprehension 带嵌套循环 ...

  3. python列表解析式如何使用_python列表推导式操作解析

    python列表推导式操作解析 这篇文章主要介绍了python列表推导式操作解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 基本格式 ​[表达式 ...

  4. Python列表解析式,还支持异步?你知道吗?

    摘要 PEP-492 和 PEP-525 通过 async/await 语法,引入了对原生协程和异步生成器的支持.本 pep 提议给列表.集合.字典解析式和生成器表达式添加异步的版本. 基本原理和目标 ...

  5. python 列表解析式_python列表解析式,字典解析式,集合解析式和生成器

    一.列表解析式(列表推倒式): 功能:是提供一种方便的列表创建方法,所以,列表解析式返回的是一个列表. 1 lst = [1, 3, 5, 8, 10] 2 ll = [x+x for x in ls ...

  6. 超详细,Python列表解析式到底该怎么用?

    Python 是一种极其多样化和强大的编程语言!当需要解决一个问题时,它有着不同的方法. 在本文中,将会展示列表解析式(List Comprehension).我们将讨论如何使用它?什么时候该或不该使 ...

  7. python列表解析式如何使用_python列表解析式,生成器,及部分内建函数使用方法...

    标准库datetime 需要导入datetime模块: import datetime datetime模块: 对日期,时间,时间戳的处理 datetime的类方法 today  () now(tz= ...

  8. [python] 列表解析式的高效与简洁

    方法一(列表解析式): list1 = ["abc","efg","hij"] list2 = [i[0] for i in list1] ...

  9. Python - 列表解析式/生成器表达式

    列表解析式: [expr for iter_var in iterable if cond_expr] 生成器表达式: (expr for iter_var in iterable if cond_e ...

  10. Python进阶3——列表解析式和生成器表达式

    列表解析式相比循环更加简介易读 abc='abcdef' codes1=[] for code in abc:codes1.append(ord(code))#返回字符对应的十进制数.codes2=[ ...

最新文章

  1. C++中的两种绑定方式(静态绑定、动态绑定)
  2. Nature Methods:快速准确的微生物来源追溯工具FEAST
  3. LeetCode-笔记-199. 二叉树的右视图——BFS广度优先搜索
  4. 电池pack结构_PACK仿真电池分析新能源汽车整车碰撞
  5. 05_ClickHouse、MergeTree系列引擎概述与存储结构、建表模板、建表语句、MergeTree设置、建表示例、数据存储、数据片段(data part)
  6. PHP中使用RabiitMQ---各项参数的使用方法
  7. SQL Server CDC配合Kafka Connect监听数据变化
  8. mysql markdown_mysql+数据库学习笔记(markdown)
  9. Python——生成一个大文件ips.txt,要求1200行 ,每行随机为172.25.254.0/24段的ip(京东二面笔试题)
  10. IBM\浪潮\Dell\HP服务器raid(磁盘阵列)配置
  11. ZooKeeper入门(二)
  12. python之用yagmail模块发送邮件
  13. struts+spring action应配置为scope=prototype
  14. 深入浅出 Cocoa 之 Core Data(3)- 使用绑定
  15. 中望cad文字显示问号怎么办_如果CAD工程图显示乱码怎么办?
  16. layui日期插件样式冲突
  17. Le le's picture when aged 5 monthes old_拔剑-浆糊的传说_新浪博客
  18. 人生路上前进的方向----有几条线贯穿自己的人生
  19. 开发团队建设与管理的一些心得
  20. Android--CoordinatorLayout源码分析及NestedScrolling机制

热门文章

  1. 本地安全管理审核UDF 第2版 - lsasecur.au3
  2. 关于B树的思考:m阶B树的非根非叶节点为什么要至少为ceil(m/2)个孩子? c/c++描述
  3. 西湖论剑 web 5 easyjson
  4. 2021-06-31 rockchip rv1126编译说明
  5. 通过get和post访问HTTP接口的方法
  6. Redis配置详解(转)
  7. 老九学堂 学习 C++ 第四天
  8. 利用开源工具搭一套汉英翻译系统(二):词对齐
  9. ansible meta目录
  10. 网络编程——Java Email