Python推导式使用详情
推导式
什么是推导式
推导式是 for 循环的简化使用方法,使用推导式,将一个可迭代对象中的数据遍历到某一个容器当中。简单的来说就是用一行for循环语句,遍历一个可迭代对象中的所有数据,然后将遍历出来的数据进行处理放入对应的容器中的一个过程和方式。
和推导类似作用的还有三元运算符,三元运算符是条件判断语句的简化使用方法。
语法:
val for val in Iterable
就是存入容器中的数据
+ for循环语句
推导式有三种表达方式,分别用对应的符号包裹推导式语句:
- 列表推导试:[val for val in Iterable]
- 集合推导式:{val for val in Iterable}
- 字典推导式:{x,y for x,y in Iterable}
列表推导式:
列表推到式,遍历出来的数据最终就会变成一个列表数据。
基本语法
列表中存入10条数据:
# 常规写法 lst = [] for i in range(1, 11): lst.append(i) print(lst) # [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] # 推导式写法 lst = [i for i in range(1, 11)] print(lst) # [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
其它使用方法
单循环推导式:
# 处理容器中的数据:[1, 2, 3, 4, 5] -> [3, 6, 9, 12, 15] lst = [1, 2, 3, 4, 5] # 普通写法 new_lst = [] for i in lst: res = i * 3 new_lst.append(res) print(new_lst) # [3, 6, 9, 12, 15] # 推导式写法 new_lst = [i * 3 for i in lst] print(new_lst) # [3, 6, 9, 12, 15]
带有判断条件的单循环推导式:
# 过滤出奇数 lst = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9] # 普通写法 new_lst = [] for i in lst: if i % 2 == 1: new_lst.append(i) print(new_lst) # [1, 3, 5, 7, 9] # 推导式写法 # 推导式使用单项分支只能是在for语句结束之后使用 new_lst = [i for i in lst if i % 2 == 1] print(new_lst) # [1, 3, 5, 7, 9]
多循环推导式:
# 两个列表中的数据相加求和 lst = [1, 2, 3] lst1 = [11, 22, 33] # 普通方法 new_lst = [] for i in lst: for j in lst1: res = i + j new_lst.append(res) print(new_lst) # [12, 23, 34, 13, 24, 35, 14, 25, 36] # 推导式写法 new_lst = [i + j for i in lst for j in lst1] print(new_lst) # [12, 23, 34, 13, 24, 35, 14, 25, 36]
列表推导式练习题
- 1、将字典中的数据变成['x=A', 'y=B', 'z=c']的样式
{'x': 'A', 'y': 'B', 'z': 'C' }
- 2、将所用元素变成纯小写
["ADDD","dddDD","DDaa","sss"]
- 3、x是0-5之间的偶数,y是0-5之间的奇数 把x,y组成一起变成元组,放到列表当中
- 4、使用列表推导式 制作所有99乘法表中的运算
- 5、求M,N中矩阵和元素的乘积
M = [[1,2,3], [4,5,6], [7,8,9]]
N = [[2,2,2], [3,3,3], [4,4,4]]
# 第五题解法之一 # =>实现效果1 [2, 4, 6, 12, 15, 18, 28, 32, 36] # =>实现效果2 [[2, 4, 6], [12, 15, 18], [28, 32, 36]] # 实现效果 1 lst_new = [] for i in range(len(M)) : for j in range(len(N)) : res = M[i][j] * N[i][j] lst_new.append(res) print(lst_new) # 推导式写法 res = [M[i][j]*N[i][j] for i in range(len(M)) for j in range(len(N))] print(res) # 实现效果 2 lst_new = [] for i in range(len(M)) : lst_new2 = [] for j in range(len(N)) : res = M[i][j] * N[i][j] lst_new2.append(res) lst_new.append(lst_new2) print(lst_new) # 推导式写法 res = [[M[i][j]*N[i][j] for j in range(len(M))] for i in range(len(N))] print(res)
集合推导式
集合推导式和列表推导式的用法基本一样,但是外面使用大括号包括,得到的数据是一个集合。
例题:
''' 案例: 满足年龄在18到21,存款大于等于5000,小于等于5500的人 开卡格式为:尊贵VIP卡老X(姓氏),否则开卡格式为:抠脚大汉老X(姓氏) 把开卡的种类统计出来 ''' lst = [ {"name": "刘鑫炜", "age": 18, "money": 10000}, {"name": "刘聪", "age": 19, "money": 5100}, {"name": "刘子豪", "age": 20, "money": 4800}, {"name": "孔祥群", "age": 21, "money": 2000}, {"name": "宋云杰", "age": 18, "money": 20} ] # 常规写法 setvar = set() for i in lst: if (18 <= i['age'] <= 21) and (5000 <= i['money'] <= 5500): res = '尊贵VIP老' + i['name'][0] else: res = '抠脚老汉' + i['name'][0] setvar.add(res) print(setvar) # {'尊贵VIP老刘', '抠脚老汉刘', '抠脚老汉孔', '抠脚老汉宋'} # 打印显示只有4个元素,是因为集合的自动去重 # 使用集合推导式 # 推导式只能使用单项分支,但是可以在返回值使用三元运算符 setvar = { "尊贵VIP卡老" + i["name"][0] if 18 <= i["age"] <= 21 and 5000 <= i["money"] <= 5500 else "抠脚大汉卡老" + i["name"][0] for i in lst} print(setvar) # {'抠脚大汉卡老孔', '抠脚大汉卡老刘', '尊贵VIP卡老刘', '抠脚大汉卡老宋'}
字典推导式
字典推导式也是一样的用法,但是字典的数据是以键值对的形式存在的,所以返回的数据、或者要将返回的数据变成两个,以对应键值。
基础语法:
将列表中的键值对的变成一个字典
lst = [{'A': 'a'}, {'B': 'b'}] dct = {k:v for i in lst for k,v in i.items()} print(dct) # {'A': 'a', 'B': 'b'}
字典推导式常用以配合的函数
函数 | 作用 |
---|---|
enumerate | 枚举,根据索引号码将可迭代对象中的值一一配对成元组,返回迭代器。 |
zip | 将多个可迭代对象中的值一一对应组成元组,返回迭代器。 |
enumerate
功能:
枚举,根据索引号码 和 Iterable 中的值,一个一个拿出来配对组成元组,放入迭代器中,然后返回迭代器。
语法:
enumerate(iterable, [start = 0])
参数:
iterable:可迭代数据
start:可以选择开始的索引号(默认从0开始索引)
基本语法:
from collections import Iterator lst = ['东', '南', '西', '北'] # 基本使用 it = enumerate(lst) # 实现功能返回迭代器 print(isinstance(it, Iterator)) # True # 强转成列表 new_lst = list(it) print(new_lst) # [(0, '东'), (1, '南'), (2, '西'), (3, '北')] """ 可以看到里面的元列表中的数据和对应的索引号码一一对应成了元组 """
上面的举例当中,如果使用字典推导式和enumerate函数配合,就可以用一句话达成组成一个字典的目的。
from collections import Iterator lst = ['东', '南', '西', '北'] # enumerate 配合使用字典推导式 变成字典 dct = {k: v for k, v in enumerate(lst)} print(dct) # {0: '东', 1: '南', 2: '西', 3: '北'}
zip
功能:
将多个Iterable中的值,一个一个拿出来配对组成元组放入迭代器中,如果某个元素多出,没有匹配项就会被舍弃。
语法:
zip(iterable, iterable1, ……)
参数就是一个个的可迭代对象。
基本语法:
下面的举例当中,将三个列表中的元素以一一对应组成元组,但是最小的列表中只有三个元素,所以只能一一对应组成三对元组,而多出的元素就被舍弃。
lst1 = [1, 2, 3, 4, 5] lst2 = ['a', 'b', 'c', 'd'] lst3 = ['A', 'B', 'C'] it = zip(lst1, lst2, lst3) lst = list(it) print(lst) # [(1, 'a', 'A'), (2, 'b', 'B'), (3, 'c', 'C')]
为什么没有元组推导式?
推导式我们到此学习完了,但是我们就发现,推导式就是在容器中使用一个for循环而已,为什么没有元组推导式?
优先使用推导式
在有的时候,我们需要在一个列表或者其它的一些容器中存放大量的值,我们一般会怎么使用呢?比如在初始化一个列表的时候我们使用for循环和append的方法去创建吗?
这里大家注意,如果条件允许的话,那么我们一定是要优先使用推导式而不是for循环加append的方式,原因很简单,因为底层逻辑的不同,使推导式的执行速度相比for循环加append更快。
import time # 列表循环插入数据 start_time = time.perf_counter() lst = [] for i in range(15000000): lst.append(i) end_time = time.perf_counter() print(end_time - start_time) # 1.7453036000000002 """ 推导式比循环速度更快 """ start_time = time.perf_counter() new_lst1 = [i for i in range(15000000)] end_time = time.perf_counter() print(end_time - start_time) # 0.7337192000000001
经过测试我们可以看到,推导式的速度大约是for循环的2倍多,是什么导致的?还记得我们之前使用过的dis
模块吗?
import dis def loop(): lst = [] for i in range(10): lst.append(i) return lst def der(): lst = [i for i in range(10)] return lst dis.dis(loop) print('-' * 100) dis.dis(der)
结果如下:
4 0 BUILD_LIST 0
2 STORE_FAST 0 (lst)
5 4 SETUP_LOOP 26 (to 32)
6 LOAD_GLOBAL 0 (range)
8 LOAD_CONST 1 (10)
10 CALL_FUNCTION 1
12 GET_ITER
>> 14 FOR_ITER 14 (to 30)
16 STORE_FAST 1 (i)
6 18 LOAD_FAST 0 (lst)
20 LOAD_ATTR 1 (append)
22 LOAD_FAST 1 (i)
24 CALL_FUNCTION 1
26 POP_TOP
28 JUMP_ABSOLUTE 14
>> 30 POP_BLOCK
7 >> 32 LOAD_FAST 0 (lst)
34 RETURN_VALUE
-----------------------------------------------------------------------------
11 0 LOAD_CONST 1 (<code object <listcomp> at 0x000002C71AD950C0, file "tset.py", line 11>)
2 LOAD_CONST 2 ('der.<locals>.<listcomp>')
4 MAKE_FUNCTION 0
6 LOAD_GLOBAL 0 (range)
8 LOAD_CONST 3 (10)
10 CALL_FUNCTION 1
12 GET_ITER
14 CALL_FUNCTION 1
16 STORE_FAST 0 (lst)
12 18 LOAD_FAST 0 (lst)
20 RETURN_VALUE
从上述结果中我们就是可以看出,在这种情况下,for循环因为开始定义列表、循环中的append方法的是使用,比推导式要多出几个环节,因此速度相比之下变得很慢,这就是原因。
再次我们也再说一句,之后碰到关于速度和底层之类的疑惑的时候,就可以简单的使用timeit
和dis
去验证和简答的理解。
到此这篇关于Python推导式使用详情的文章就介绍到这了,更多相关Python推导式内容请搜索猪先飞以前的文章或继续浏览下面的相关文章希望大家以后多多支持猪先飞!
原文出处:https://blog.51cto.com/msr20666/5285101
相关文章
- 这篇文章主要介绍了python-opencv-画外接矩形框的实例代码,代码简单易懂,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-09-04
Python astype(np.float)函数使用方法解析
这篇文章主要介绍了Python astype(np.float)函数使用方法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下...2020-06-08- 2022虎年新年即将来临,小编为大家带来了一个利用Python编写的虎年烟花特效,堪称全网最绚烂,文中的示例代码简洁易懂,感兴趣的同学可以动手试一试...2022-02-14
- 在本篇文章里小编给大家分享的是一篇关于python中numpy.empty()函数实例讲解内容,对此有兴趣的朋友们可以学习下。...2021-02-06
- 有时为了网站安全和版权问题,会对自己写的php源码进行加密,在php加密技术上最常用的是zend公司的zend guard 加密软件,现在我们来图文讲解一下。 下面就简单说说如何...2016-11-25
python-for x in range的用法(注意要点、细节)
这篇文章主要介绍了python-for x in range的用法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-05-10- 这篇文章主要介绍了Python 图片转数组,二进制互转操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-09
- 这篇文章主要介绍了Python中的imread()函数用法说明,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-16
- 这篇文章主要介绍了python如何实现b站直播自动发送弹幕,帮助大家更好的理解和学习使用python,感兴趣的朋友可以了解下...2021-02-20
- ps软件是现在很多人都会使用到的,HSL面板在ps软件中又有着非常独特的作用。这次文章就给大家介绍下ps怎么使用HSL面板,还不知道使用方法的下面一起来看看。  ...2017-07-06
python Matplotlib基础--如何添加文本和标注
这篇文章主要介绍了python Matplotlib基础--如何添加文本和标注,帮助大家更好的利用Matplotlib绘制图表,感兴趣的朋友可以了解下...2021-01-26- 这篇文章主要介绍了解决python 使用openpyxl读写大文件的坑,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-13
- 今天小编就为大家分享一篇python 计算方位角实例(根据两点的坐标计算),具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-04-27
- 许多的朋友对于Plesk控制面板应用不是非常的了解特别是英文版的Plesk控制面板,在这里小编整理了一些关于Plesk控制面板常用的使用方案整理,具体如下。 本文基于Linu...2016-10-10
- 这篇文章主要为大家详细介绍了python实现双色球随机选号,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-05-02
- 在本篇文章里小编给大家整理的是一篇关于python中使用np.delete()的实例方法,对此有兴趣的朋友们可以学习参考下。...2021-02-01
使用insertAfter()方法在现有元素后添加一个新元素
复制代码 代码如下: //在现有元素后添加一个新元素 function insertAfter(newElement, targetElement){ var parent = targetElement.parentNode; if (parent.lastChild == targetElement){ parent.appendChild(newEl...2014-05-31- 这篇文章主要介绍了使用Python的pencolor函数实现渐变色功能,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-03-09
- 大概有如下步骤 新建项目Bejs 新建文件package.json 新建文件Gruntfile.js 命令行执行grunt任务 一、新建项目Bejs源码放在src下,该目录有两个js文件,selector.js和ajax.js。编译后代码放在dest,这个grunt会...2014-06-07
使用percona-toolkit操作MySQL的实用命令小结
1.pt-archiver 功能介绍: 将mysql数据库中表的记录归档到另外一个表或者文件 用法介绍: pt-archiver [OPTION...] --source DSN --where WHERE 这个工具只是归档旧的数据,不会对线上数据的OLTP查询造成太大影响,你可以将...2015-11-24