解析式
其目的主要用来减少编程行数,并减少栈帧从而达到代码优化的效果
In [6]: [i ** 2 for i in range(11)]
Out[6]: [0, 1, 4, 9, 16, 25, 36, 49, 64, 81, 100]
将生产环节元素表达式放在最前面
列表解析语法
[返回值 for 元素 可迭代对象if条件]
使用中括号表示,内部for为循环跟if条件可选,返回一个新的列表
这样可以简化编程中书写并且减少了栈帧,从而达到优化效果
例:对i进行取模
In [23]: [ i for i in range(1,10) if i % 2 == 0]
Out[23]: [2, 4, 6, 8]
解析式中不能使用else 和 elif 所以需要用or 和 and来代替
In [4]: [i for i in range(20) if i %2 == 0 and i%3 == 0]
Out[4]: [0, 6, 12, 18]
使用or/ not
In [5]: [i for i in range(20) if i %2 == 0 or noti % 2== 0]
Out[5]: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11,12, 13, 14, 15, 16, 17, 18, 19]
使用and / not
In [11]: [i for i in range(20) if i % 2 ==0 andnot i % 3]
Out[11]: [0, 6, 12, 18]
可以理解为:
In [13]: [i for i in range(20) if i % 2 ==0 andnot i % 3 != 0]
Out[13]: [0, 6, 12, 18]
推导式的多重过滤
使用and进行代替多个if:if:if,对多重邓加进行代替
for i in iter1:
for j in iter2:
lst.append(expr)
等价于:
In[14]: [expr for i in iter1 for j in iter2]
例:
1.
In [14]: [ (x,y) for x in 'abc' for y in range(3)]
Out[14]:
[('a', 0),
('a', 1),
('a', 2),
('b', 0),
('b', 1),
('b', 2),
('c', 0),
('c', 1),
('c', 2)]
等价于:
In [21]: for i in 'abc':
...: fory in range(3):
...: lst.append((i,y))
In [26]: [{x:y} for x in "abcde" for yin range(3)]
Out[26]:
[{'a': 0},
{'a': 1},
{'a': 2},
{'b': 0},
{'b': 1},
{'b': 2},
{'c': 0},
等价于:
In [34]: for i in "abc":
...: forx in range(3):
...: lst.append({i:x})
在语句上第三个最方便,但是第一条最容易理解:
In [36]: [ (i,j) for i in range(7) if i > 4for j in range(20,25) if j > 23 ]
Out[36]: [(5, 24), (6, 24)]
In [37]: [(i,j) for i in range(7) for j inrange(20,25) if i > 4 if j > 23]
Out[37]: [(5, 24), (6, 24)]
In [39]: [(i,j) for i in range(7) for j inrange(20,25) if i > 4 and j > 23]
Out[39]: [(5, 24), (6, 24)]
求99乘法表
[print ('{}*{}={:<3}{}'.format(j,i,i*j,'\n' ifi == j else ''),end = "") for i in range(1,10) for j in range(1,i+1)]
生成器表达式
与解析式一样,只不过将中括号改为小括号
In [2]: a = (n for n in range(10))
In [3]: a
Out[3]: <generator object <genexpr> at 0x7fed4ea1cfc0>
In [5]: next(a)
Out[5]: 0
In [6]: next(a)
Out[6]: 1
In [7]: next(a)
Out[7]: 2
迭代中,只能使用一次,当获取后则直接将其销毁,不再保留至内存空间中
使用next(a)的方法,必须一定是可迭代对象
使用生成器的好处:
·延迟计算
·返回迭代器,可以进行迭代
·从签到后走完一遍,不能回头
例:
使用print进行观察
In [8]: a = (print("{}".format(i+1)) for i in range(2))
In [9]: print(a)
<generator object <genexpr> at0x7fed51641f10>
In [10]: b = next(a)
1
In [11]: b = next(a)
2
In [12]: b = next(a)
---------------------------------------------------------------------------
StopIteration Traceback(most recent call last)
<ipython-input-12-78fca016cc6c> in<module>()
----> 1 b = next(a)
StopIteration:
生成器和列表解析式对比
立即生成和延后计算,可以嵌套在列表解析式中从返回值来讲,更节省内存,生成器则全部生成并返回
生成器没有数据,占用内存极少,使用的时候逐个返回
列表需要占用更多的内存
计算速度
生成器耗时间非常短,列表解析消耗时间略长
生成器本身是一个迭代器
速度对比
In[14]: %timeit [x for x in range(500)]
18 μs ± 102 ns per loop (mean± std. dev. of 7 runs, 100000 loops each)
In[15]: %timeit (x for x in range(500))
741 ns± 7.58 ns perloop (mean ± std. dev. of 7 runs, 1000000 loops each)
集合解析式
将返回值for作为可迭代对象通过if条件换位,
将括号换至为大伙靠,生成后立刻返回一个集合
In [19]: {(x,x+1) for x in range(10)}
Out[19]:
{(0, 1),
(1, 2),
(2, 3),
(3, 4),
(4, 5),
(5, 6),
(6, 7),
(7, 8),
(8, 9),
(9, 10)}
字典解析式:
In [25]: {chr(0x41+x):x**2 for x in range(10) }
Out[25]:
{'A': 0,
'B': 1,
'C': 4,
'D': 9,
'E': 16,
'F': 25,
'G': 36,
'H': 49,
'I': 64,
'J': 81}
In [26]: {str(x):y for x in range(3) for y inrange(4)}
Out[26]: {'0': 3, '1': 3, '2': 3}
内建函数
iter 将一个可迭代对象封装为一个迭代器
a = iter(range(5))
迭代器对象,迭代器本身是可迭代的,所以说可以通过iter方法将可迭代对象封装为迭代器然后通过next方法进行迭代
zip 拉链函数
将多个可迭代函数合并在一起,返回一个迭代器,将每次不同对象中取到的元素合并到一个元组中
In [40]: list(zip(range(10),range(10)))
Out[40]:
[(0, 0),
(1, 1),
(2, 2),
(3, 3),
(4, 4),
(5, 5),
(6, 6),
(7, 7),
(8, 8),
(9, 9)]
zip为木桶原理,取当前最短的,并不再继续
In [41]: list(zip(range(10),range(10),range(5)))
Out[41]: [(0, 0, 0), (1, 1, 1), (2, 2, 2), (3, 3,3), (4, 4, 4)]
如果某一个值超出,则只选择最短的值,不会再进行多次选取
In [42]: list(zip(range(10),range(10),range(50)))
Out[42]:
[(0, 0, 0),
(1, 1, 1),
(2, 2, 2),
(3, 3, 3),
(4, 4, 4),
(5, 5, 5),
(6, 6, 6),
(7, 7, 7),
(8, 8, 8),
(9, 9, 9)]
In [43]: list(zip(range(10),range(10),range(3)))
Out[43]: [(0, 0, 0), (1, 1, 1), (2, 2, 2)]