介绍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输出一个杨辉三角的例子
Jun 13 Python
python递归全排列实现方法
Aug 18 Python
对python实现合并两个排序链表的方法详解
Jan 23 Python
python爬虫 2019中国好声音评论爬取过程解析
Aug 26 Python
Python3显示当前时间、计算时间差及时间加减法示例代码
Sep 07 Python
Python实现密码薄文件读写操作
Dec 16 Python
python实现12306登录并保存cookie的方法示例
Dec 17 Python
Python requests设置代理的方法步骤
Feb 23 Python
python使用pandas抽样训练数据中某个类别实例
Feb 28 Python
Django 解决由save方法引发的错误
May 21 Python
用python查找统一局域网下ip对应的mac地址
Jan 13 Python
详解Python openpyxl库的基本应用
Feb 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的chr和ord函数实现字符加减乘除运算实现代码
2011/12/05 PHP
PHP使用mysqldump命令导出数据库
2015/04/14 PHP
JavaScript Undefined,Null类型和NaN值区别
2008/10/22 Javascript
jquery高级编程的最佳实践详解
2014/03/23 Javascript
javascript使用switch case实现动态改变超级链接文字及地址
2014/12/16 Javascript
JavaScript中的依赖注入详解
2015/03/18 Javascript
jQuery控制网页打印指定区域的方法
2015/04/07 Javascript
浅谈javascript事件取消和阻止冒泡
2015/05/26 Javascript
Javascript 计算字符串在localStorage中所占字节数
2015/10/21 Javascript
jQuery实现横向带缓冲的水平运动效果(附demo源码下载)
2016/01/29 Javascript
使用JavaScript实现ajax的实例代码
2016/05/11 Javascript
AngularJS实现一次监听多个值发生的变化
2016/08/31 Javascript
BootStrap tab选项卡使用小结
2020/08/09 Javascript
javascript入门之window对象【新手必看】
2016/11/22 Javascript
基于JavaScript实现多级菜单效果
2017/07/25 Javascript
对Vue2 自定义全局指令Vue.directive和指令的生命周期介绍
2018/08/30 Javascript
vue-自定义组件传值的实例讲解
2018/09/18 Javascript
Layer.js实现表格溢出内容省略号显示,悬停显示全部的方法
2019/09/16 Javascript
原生js中运算符及流程控制示例详解
2021/01/05 Javascript
python使用win32com在百度空间插入html元素示例
2014/02/20 Python
python 执行shell命令并将结果保存的实例
2018/05/11 Python
浅谈python3.6的tkinter运行问题
2019/02/22 Python
python保存字典和读取字典的实例代码
2019/07/07 Python
django 快速启动数据库客户端程序的方法示例
2019/08/16 Python
解决IDEA 的 plugins 搜不到任何的插件问题
2020/05/04 Python
Python RabbitMQ实现简单的进程间通信示例
2020/07/02 Python
Python基于callable函数检测对象是否可被调用
2020/10/16 Python
巴基斯坦购物网站:Goto
2019/03/11 全球购物
澳大利亚网上买书:Angus & Robertson
2019/07/21 全球购物
军训感想500字
2014/02/20 职场文书
幼儿园中班教师寄语
2014/04/03 职场文书
单位介绍信格式
2015/01/31 职场文书
美德少年事迹材料(2016推荐版)
2016/02/25 职场文书
golang中的并发和并行
2021/05/08 Golang
一篇文章带你复习java知识点
2021/06/28 Java/Android
java实现面板之间切换功能
2022/06/10 Java/Android