介绍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中列表(list)操作方法汇总
Aug 18 Python
Python3实现的Mysql数据库操作封装类
Jun 06 Python
对TensorFlow的assign赋值用法详解
Jul 30 Python
5分钟 Pipenv 上手指南
Dec 20 Python
Python txt文件加入字典并查询的方法
Jan 15 Python
python在openstreetmap地图上绘制路线图的实现
Jul 11 Python
python装饰器练习题及答案
Nov 01 Python
在 Linux/Mac 下为Python函数添加超时时间的方法
Feb 20 Python
pandas使用之宽表变窄表的实现
Apr 12 Python
python实现控制台输出颜色
Mar 02 Python
简述python四种分词工具,盘点哪个更好用?
Apr 13 Python
Python利器openpyxl之操作excel表格
Apr 17 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依赖注入和控制反转
2016/05/11 PHP
pjblog中的UBBCode.js
2007/04/25 Javascript
jquery移动listbox的值原理及代码
2013/05/03 Javascript
浅析document.createDocumentFragment()与js效率
2013/07/08 Javascript
JS读取XML文件示例代码
2013/11/15 Javascript
jquery的each方法使用示例分享
2014/03/25 Javascript
jQuery实现鼠标经过图片预览大图效果
2014/04/10 Javascript
js处理php输出时间戳对不上号的解决方法
2014/06/20 Javascript
js的回调函数详解
2015/01/05 Javascript
Javascript核心读书有感之语句
2015/02/11 Javascript
JS实现网站菜单拖拽移位效果的方法
2015/09/24 Javascript
jQuery 检查某个元素在页面上是否存在实例代码
2016/10/27 Javascript
浅谈jquery页面初始化的4种方式
2016/11/27 Javascript
详解基于node的前端项目编译时内存溢出问题
2017/08/01 Javascript
vue系列之动态路由详解【原创】
2017/09/10 Javascript
详解Angular5路由传值方式及其相关问题
2018/04/28 Javascript
JS中appendChild追加子节点无效的解决方法
2018/10/14 Javascript
JavaScript基于遍历操作实现对象深拷贝功能示例
2019/03/05 Javascript
javascript自定义日期比较函数用法示例
2019/07/22 Javascript
使用Vue-cli3.0创建的项目 如何发布npm包
2019/10/10 Javascript
微信小程序实现录音功能
2019/11/22 Javascript
vue-router懒加载的3种方式汇总
2021/02/28 Vue.js
python编程实现归并排序
2017/04/14 Python
python使用mysql数据库示例代码
2017/05/21 Python
python ddt实现数据驱动
2018/03/14 Python
python简单操作excle的方法
2018/09/12 Python
搞清楚 Python traceback的具体使用方法
2019/05/13 Python
从0到1使用python开发一个半自动答题小程序的实现
2020/05/12 Python
浅谈Python程序的错误:变量未定义
2020/06/02 Python
calendar在python3时间中常用函数举例详解
2020/11/18 Python
泰国网上购物:Shopee泰国
2018/09/14 全球购物
马耳他航空公司官方网站:Air Malta
2019/05/15 全球购物
史上最全面的Java面试题汇总!
2015/02/03 面试题
给医务人员表扬信
2014/01/12 职场文书
大学生党课心得体会
2016/01/07 职场文书
MySQL单表千万级数据处理的思路分享
2021/06/05 MySQL