介绍Python中内置的itertools模块


Posted in Python onApril 29, 2015

Python的内建模块itertools提供了非常有用的用于操作迭代对象的函数。

首先,我们看看itertools提供的几个“无限”迭代器:

>>> import itertools
>>> natuals = itertools.count(1)
>>> for n in natuals:
...   print n
...
1
2
3
...

因为count()会创建一个无限的迭代器,所以上述代码会打印出自然数序列,根本停不下来,只能按Ctrl+C退出。

cycle()会把传入的一个序列无限重复下去:

>>> import itertools
>>> cs = itertools.cycle('ABC') # 注意字符串也是序列的一种
>>> for c in cs:
...   print c
...
'A'
'B'
'C'
'A'
'B'
'C'
...

同样停不下来。

repeat()负责把一个元素无限重复下去,不过如果提供第二个参数就可以限定重复次数:

>>> ns = itertools.repeat('A', 10)
>>> for n in ns:
...   print n
...

打印10次'A'

无限序列只有在for迭代时才会无限地迭代下去,如果只是创建了一个迭代对象,它不会事先把无限个元素生成出来,事实上也不可能在内存中创建无限多个元素。

无限序列虽然可以无限迭代下去,但是通常我们会通过takewhile()等函数根据条件判断来截取出一个有限的序列:

>>> natuals = itertools.count(1)
>>> ns = itertools.takewhile(lambda x: x <= 10, natuals)
>>> for n in ns:
...   print n
...

打印出1到10

itertools提供的几个迭代器操作函数更加有用:
chain()

chain()可以把一组迭代对象串联起来,形成一个更大的迭代器:

for c in chain('ABC', 'XYZ'):
  print c
# 迭代效果:'A' 'B' 'C' 'X' 'Y' 'Z'

groupby()

groupby()把迭代器中相邻的重复元素挑出来放在一起:

>>> for key, group in itertools.groupby('AAABBBCCAAA'):
...   print key, list(group) # 为什么这里要用list()函数呢?
...
A ['A', 'A', 'A']
B ['B', 'B', 'B']
C ['C', 'C']
A ['A', 'A', 'A']

实际上挑选规则是通过函数完成的,只要作用于函数的两个元素返回的值相等,这两个元素就被认为是在一组的,而函数返回值作为组的key。如果我们要忽略大小写分组,就可以让元素'A'和'a'都返回相同的key:

>>> for key, group in itertools.groupby('AaaBBbcCAAa', lambda c: c.upper()):
...   print key, list(group)
...
A ['A', 'a', 'a']
B ['B', 'B', 'b']
C ['c', 'C']
A ['A', 'A', 'a']

imap()

imap()和map()的区别在于,imap()可以作用于无穷序列,并且,如果两个序列的长度不一致,以短的那个为准。

>>> for x in itertools.imap(lambda x, y: x * y, [10, 20, 30], itertools.count(1)):
...   print x
...
10
40
90

注意imap()返回一个迭代对象,而map()返回list。当你调用map()时,已经计算完毕:

>>> r = map(lambda x: x*x, [1, 2, 3])
>>> r # r已经计算出来了
[1, 4, 9]

当你调用imap()时,并没有进行任何计算:

>>> r = itertools.imap(lambda x: x*x, [1, 2, 3])
>>> r
<itertools.imap object at 0x103d3ff90>
# r只是一个迭代对象

必须用for循环对r进行迭代,才会在每次循环过程中计算出下一个元素:

>>> for x in r:
...   print x
...
1
4
9

这说明imap()实现了“惰性计算”,也就是在需要获得结果的时候才计算。类似imap()这样能够实现惰性计算的函数就可以处理无限序列:

>>> r = itertools.imap(lambda x: x*x, itertools.count(1))
>>> for n in itertools.takewhile(lambda x: x<100, r):
...   print n
...

结果是什么?

如果把imap()换成map()去处理无限序列会有什么结果?

>>> r = map(lambda x: x*x, itertools.count(1))

结果是什么?

ifilter()

不用多说了,ifilter()就是filter()的惰性实现。
小结

itertools模块提供的全部是处理迭代功能的函数,它们的返回值不是list,而是迭代对象,只有用for循环迭代的时候才真正计算。

Python 相关文章推荐
python选择排序算法的实现代码
Nov 21 Python
Python批量重命名同一文件夹下文件的方法
May 25 Python
Python使用装饰器进行django开发实例代码
Feb 06 Python
django静态文件加载的方法
May 20 Python
mac下如何将python2.7改为python3
Jul 13 Python
python生成九宫格图片
Nov 19 Python
Python3多线程版TCP端口扫描器
Aug 31 Python
基于Tensorflow使用CPU而不用GPU问题的解决
Feb 07 Python
Python虚拟环境库virtualenvwrapper安装及使用
Jun 17 Python
python删除csv文件的行列
Apr 06 Python
用Python提取PDF表格的方法
Apr 11 Python
浅析Django接口版本控制
Jun 26 Python
python使用fileinput模块实现逐行读取文件的方法
Apr 29 #Python
python将字符串转换成数组的方法
Apr 29 #Python
Python中使用hashlib模块处理算法的教程
Apr 28 #Python
简单介绍Python中的struct模块
Apr 28 #Python
在Python中使用base64模块处理字符编码的教程
Apr 28 #Python
使用Python的内建模块collections的教程
Apr 28 #Python
进一步探究Python中的正则表达式
Apr 28 #Python
You might like
PHP操作mysql函数详解,mysql和php交互函数
2011/05/19 PHP
Zend的AutoLoad机制介绍
2012/09/27 PHP
php单例模式实现(对象只被创建一次)
2012/12/05 PHP
PHP利用str_replace防注入的方法
2013/11/10 PHP
PHP函数microtime()用法与说明
2013/12/04 PHP
php生成固定长度纯数字编码的方法
2015/07/09 PHP
Symfony2学习笔记之控制器用法详解
2016/03/17 PHP
PHP如何获取当前主机、域名、网址、路径、端口等参数
2017/06/09 PHP
js+CSS 图片等比缩小并垂直居中实现代码
2008/12/01 Javascript
js返回上一页并刷新代码整理
2012/12/21 Javascript
简单的jquery左侧导航栏和页面选中效果
2014/08/21 Javascript
js图片轮播手动切换效果
2015/11/10 Javascript
jQuery实现的自动加载页面功能示例
2016/09/04 Javascript
JavaScript用二分法查找数据的实例代码
2017/06/17 Javascript
vue表单绑定实现多选框和下拉列表的实例
2017/08/12 Javascript
jQuery实现获取form表单内容及绑定数据到form表单操作分析
2018/07/03 jQuery
node中的cookie的具体使用
2018/09/13 Javascript
Vue中使用better-scroll实现轮播图组件
2020/03/07 Javascript
多个Vue项目部署到服务器的步骤记录
2020/10/22 Javascript
在Python中使用SimpleParse模块进行解析的教程
2015/04/11 Python
Python中的默认参数实例分析
2018/01/29 Python
python 获取文件下所有文件或目录os.walk()的实例
2018/04/23 Python
python爱心表白 每天都是浪漫七夕!
2018/08/18 Python
使用Python将Mysql的查询数据导出到文件的方法
2019/02/25 Python
PyQt5的PyQtGraph实践系列3之实时数据更新绘制图形
2019/05/13 Python
Python3中的最大整数和最大浮点数实例
2019/07/09 Python
Python-Seaborn热图绘制的实现方法
2019/07/15 Python
python 如何将数据写入本地txt文本文件的实现方法
2019/09/11 Python
python 多线程中join()的作用
2020/10/29 Python
英国领先的杂志订阅网站:Magazine.co.uk
2018/01/25 全球购物
新大陆软件面试题
2016/11/24 面试题
机关作风整顿个人整改措施思想汇报
2014/09/29 职场文书
独生子女证明范本
2015/06/19 职场文书
爱国主题班会教案
2015/08/14 职场文书
Go获取两个时区的时间差
2022/04/20 Golang
React自定义hook的方法
2022/06/25 Javascript