介绍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模拟新浪微博登陆功能(新浪微博爬虫)
Dec 24 Python
python中引用与复制用法实例分析
Jun 04 Python
剖析Django中模版标签的解析与参数传递
Jul 21 Python
Python 操作文件的基本方法总结
Aug 10 Python
Python 中导入csv数据的三种方法
Nov 01 Python
解决Python中list里的中文输出到html模板里的问题
Dec 17 Python
使用GitHub和Python实现持续部署的方法
May 09 Python
对PyQt5的输入对话框使用(QInputDialog)详解
Jun 25 Python
python3.6 如何将list存入txt后再读出list的方法
Jul 02 Python
Python while循环使用else语句代码实例
Feb 07 Python
哈工大自然语言处理工具箱之ltp在windows10下的安装使用教程
May 07 Python
python tqdm库的使用
Nov 30 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
首页四格,首页五格For6.0(GBK)(UTF-8)[12种组合][9-18][版主安装测试通过]
2007/09/24 PHP
Could not load type System.ServiceModel.Activation.HttpModule解决办法
2012/12/29 PHP
PHP函数microtime()用法与说明
2013/12/04 PHP
php中__destruct与register_shutdown_function执行的先后顺序问题
2014/10/17 PHP
PHP实现二叉树的深度优先与广度优先遍历方法
2015/09/28 PHP
PHP 7.0.2 正式版发布
2016/01/08 PHP
ASP.NET jQuery 实例15 通过控件CustomValidator验证CheckBoxList
2012/02/03 Javascript
用js写了一个类似php的print_r输出换行功能
2013/02/18 Javascript
jQuery产品间断向下滚动效果核心代码
2014/05/08 Javascript
js操作css属性实现div层展开关闭效果的方法
2015/05/11 Javascript
Jquery技巧(必须掌握)
2016/03/16 Javascript
基于jQuery实现仿百度首页选项卡切换效果
2016/05/29 Javascript
AngularJS ng-app 指令实例详解
2016/07/30 Javascript
jQuery中slidedown与slideup方法用法示例
2016/09/16 Javascript
JavaScript中无法通过div.style.left获取值的解决方法
2017/02/19 Javascript
Ionic + Angular.js实现图片轮播的方法示例
2017/05/21 Javascript
vue2实现可复用的轮播图carousel组件详解
2017/11/27 Javascript
详解nodejs解压版安装和配置(带有搭建前端项目脚手架)
2018/12/06 NodeJs
[01:55]2014DOTA2国际邀请赛快报:国土生病 紧急去医院治疗
2014/07/10 DOTA
Python深入学习之闭包
2014/08/31 Python
python通过post提交数据的方法
2015/05/06 Python
深入理解Python中字典的键的使用
2015/08/19 Python
python Django框架实现自定义表单提交
2016/03/25 Python
Python中defaultdict与lambda表达式用法实例小结
2018/04/09 Python
python画双y轴图像的示例代码
2019/07/07 Python
python matplotlib:plt.scatter() 大小和颜色参数详解
2020/04/14 Python
Python爬虫之爬取淘女郎照片示例详解
2020/07/28 Python
HTML5 客户端数据库简易使用:IndexedDB
2019/12/19 HTML / CSS
荣耀俄罗斯官网:HONOR俄罗斯
2020/10/31 全球购物
Java里面Pass by value和Pass by Reference是什么意思
2016/05/02 面试题
党员群众路线剖析材料
2014/10/08 职场文书
晋江市人民政府党组群众路线教育实践活动整改方案
2014/10/25 职场文书
部门优秀员工推荐信
2015/03/24 职场文书
同学聚会通知书
2015/04/20 职场文书
使用springboot暴露oracle数据接口的问题
2021/05/07 Oracle
python中if和elif的区别介绍
2021/11/07 Python