介绍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实现apahce网站日志分析示例
Apr 02 Python
以一个投票程序的实例来讲解Python的Django框架使用
Feb 18 Python
Python中动态创建类实例的方法
Mar 24 Python
Python爬虫实现网页信息抓取功能示例【URL与正则模块】
May 18 Python
对python实现模板生成脚本的方法详解
Jan 30 Python
浅谈python编译pyc工程--导包问题解决
Mar 20 Python
python中的print()输出
Apr 12 Python
PyCharm中如何直接使用Anaconda已安装的库
May 28 Python
Python代码覆盖率统计工具coverage.py用法详解
Nov 25 Python
python 使用openpyxl读取excel数据
Feb 18 Python
Python制作表白爱心合集
Jan 22 Python
Python中的datetime包与time包包和模块详情
Feb 28 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者的疑难问答(1)
2006/10/09 PHP
IIS下配置Php+Mysql+zend的图文教程
2006/12/08 PHP
PHP 和 MySQL 开发的 8 个技巧
2007/01/02 PHP
php的array_multisort()使用方法介绍
2012/05/16 PHP
PHP stripos()函数及注意事项的分析
2013/06/08 PHP
推荐一些非常不错的javascript学习资源站点
2007/08/29 Javascript
javascript 获取图片颜色
2009/04/05 Javascript
Jquery插件之多图片异步上传
2010/10/20 Javascript
侧栏跟随滚动的简单实现代码
2013/03/18 Javascript
原生js的弹出层且其内的窗口居中
2014/05/14 Javascript
基于JS实现PHP的sprintf函数实例
2015/11/14 Javascript
基于jQuery ligerUI实现分页样式
2016/09/18 Javascript
javascript创建对象的3种方法
2016/11/02 Javascript
AngularJs中 ng-repeat指令中实现含有自定义指令的动态html的方法
2017/01/19 Javascript
JS检测是否可以访问公网服务器功能代码
2017/06/19 Javascript
import与export在node.js中的使用详解
2017/09/28 Javascript
ES6学习笔记之map、set与数组、对象的对比
2018/03/01 Javascript
nodejs中密码加密处理操作详解
2018/03/20 NodeJs
vue的.vue文件是怎么run起来的(vue-loader)
2018/12/10 Javascript
vue store之状态管理模式的详细介绍
2019/06/13 Javascript
详解vue 2.6 中 slot 的新用法
2019/07/09 Javascript
vue+eslint+vscode配置教程
2019/08/09 Javascript
推荐几个不错的console调试技巧实现
2019/12/20 Javascript
[03:03]DOTA2 2017国际邀请赛开幕战队入场仪式
2017/08/09 DOTA
用于统计项目中代码总行数的Python脚本分享
2015/04/21 Python
Python实现学生成绩管理系统
2020/04/05 Python
python3的数据类型及数据类型转换实例详解
2019/08/20 Python
浅谈pymysql查询语句中带有in时传递参数的问题
2020/06/05 Python
python调用jenkinsAPI构建jenkins,并传递参数的示例
2020/12/09 Python
HTML5 canvas标签实现刮刮卡效果
2015/04/24 HTML / CSS
顺丰快递Java软件工程师面试题
2015/07/31 面试题
幼儿园门卫制度
2014/01/29 职场文书
优秀应届生求职信
2014/06/16 职场文书
人口与计划生育目标管理责任书
2014/07/29 职场文书
青年文明号汇报材料
2014/12/23 职场文书
Python中递归以及递归遍历目录详解
2021/10/24 Python