Python中的迭代器漫谈


Posted in Python onFebruary 03, 2015

问题是在Python中进行循环的时候产生的,熟悉Python的都知道,它没有类似其它语言中的for循环, 只能通过for in的方式进行循环遍历。最典型的应用就是通过range函数产生一个列表,然后用for in进行操作,如下:

#!/usr/bin/env python

for i in range(10):

    print i

代码的意义很好理解,range会产生一个列表,用for in最这个列表进行遍历,就有和类似for(i = 0;i<n;i++)同样的效果,range函数的详解可以看这里。问题又来了,range这个对象会产生一个列表,那么这个列表的内容铁定是存放在内存当中的,当需要的循环数量太大时,是相当占用内存的, 为了统计使用range占用内存的情况,我做了6次使用,分别用range产生100,10000,100000,1000000,10000000,100000000长度的列表,然后统计内存的占用:

测试代码 占用内存

range(100) 2.0MB

range(10000) 2.2MB

range(100000) 3.8MB

range(1000000) 19.5MB

range(10000000) 168.5MB

range(100000000) 1465.8MB

可以看到,随着基数的加大,占用内存呈几何倍数增加,显然在进行大循环操作的时候,要避免使用range。

为了解决上述问题,python提供了另外一个函数xrange,这个函数和range非常相似,但是占用内存比range会小很多,相关的说明可以查看这里,经过测试,用xrange产生的对象,不管参数是多少,占用内存几乎都没有变化。问题又来了,xrange内部是如何实现的,为什么和range性能相差这么大?为了验证我的猜想,先尝试用python实现类似xrange的函数zrange:

#!/usr/bin/env python

class zrange(object):

    def __init__(self,stop):

        self.__pointer=0

        self.stop=stop

    def __iter__(self):  

        return self  

    def next(self): #python3.0中,改用__next__

        if self.__pointer  >= self.stop:

            raise StopIteration

        else:

            self.__pointer = self.__pointer + 1

            return self.__pointer-1

test = zrange(10000000)

for i in test:

    print i

运行的结果和xrange一样, 对zrange进行内存占用测试,发现和xrange一样,参数的大小对内存占用几乎没有影响。那么它和range的区别在哪里呢?

前面说到,range产生的是一个列表,而无论是自定义的zrange还是系统内置的xrange产生的都是一个对象,像xrange或者zrange产生的对象,就叫做可迭代对象, 它给外部提供了一种遍历其内部元素,而不用关心其内部实现的方法。上面zrange的实现中, 最关键的实现是建立了一个内部指针__pointer, 它记录当前的访问的位置, 下次的访问就可以通过指针的状态进行相应的操作。

Python或者其它语言中,还有很多类似通过迭代的方式访问对象内容的,如读取一个文件中的内容:

#!/usr/bin/env python

f = open('zrange.py','r')

while True:

    line = f.readline()

    if not line:

        break

    print line.strip()

f.close()

大家都知道用readline要比reandlines节省资源,其实readline和readlines就类似于xrange和range,一个是通过指针记录当前位置,下次访问把指针往前移动一个单位,另外一个是直接把所有内容存放到内存当中。文件操作函数中,还可以通过seek手动的调整指针的位置,从而达到跳过或者重复读取某些内容的目的。

可以说,迭代器的实现中,其内部指针是节省资源,让迭代正常运行的关键。

Python 相关文章推荐
Python查找相似单词的方法
Mar 05 Python
Python类方法__init__和__del__构造、析构过程分析
Mar 06 Python
Windows和Linux下使用Python访问SqlServer的方法介绍
Mar 10 Python
Python实现二叉树结构与进行二叉树遍历的方法详解
May 24 Python
Python实现常见的回文字符串算法
Nov 14 Python
python批量修改图片尺寸,并保存指定路径的实现方法
Jul 04 Python
Python代码太长换行的实现
Jul 05 Python
python-Web-flask-视图内容和模板知识点西宁街
Aug 23 Python
3行Python代码实现图像照片抠图和换底色的方法
Oct 10 Python
关于python中的xpath解析定位
Mar 06 Python
Python基于numpy模块实现回归预测
May 14 Python
python之openpyxl模块的安装和基本用法(excel管理)
Feb 03 Python
Python描述器descriptor详解
Feb 03 #Python
理解Python中的With语句
Feb 02 #Python
Linux环境下MySQL-python安装过程分享
Feb 02 #Python
Python中用pycurl监控http响应时间脚本分享
Feb 02 #Python
Python列表(list)常用操作方法小结
Feb 02 #Python
Python Sleep休眠函数使用简单实例
Feb 02 #Python
Python中实现从目录中过滤出指定文件类型的文件
Feb 02 #Python
You might like
php下用GD生成生成缩略图的两个选择和区别
2007/04/17 PHP
有关 PHP 和 MySQL 时区的一点总结
2008/03/26 PHP
简单谈谈php中的unicode和utf8编码
2015/06/10 PHP
ThinkPHP3.1.x修改成功与失败跳转页面的方法
2017/09/29 PHP
Yii框架核心组件类实例详解
2019/08/06 PHP
通用于ie和firefox的函数 GetCurrentStyle (obj, prop)
2006/12/27 Javascript
jcarousellite.js 基于Jquery的图片无缝滚动插件
2010/12/30 Javascript
JavaScript跨平台的开源框架NativeScript
2015/03/24 Javascript
js+html5实现可在手机上玩的拼图游戏
2015/07/17 Javascript
逐一介绍Jquery data()、Jquery stop()、jquery delay()函数(详)
2015/11/04 Javascript
javascript 动态脚本添加的简单方法
2016/10/11 Javascript
小程序开发实战:实现九宫格界面的导航的代码实现
2017/01/19 Javascript
在Vue中使用icon 字体图标的方法
2019/06/14 Javascript
微信小程序实现张图片合成为一张并下载
2019/07/16 Javascript
微信小程序swiper组件实现抖音翻页切换视频功能的实例代码
2020/06/24 Javascript
利用打码兔和超人打码自封装的打码类分享
2014/03/16 Python
Python基于回溯法子集树模板解决全排列问题示例
2017/09/07 Python
Python实现冒泡排序的简单应用示例
2017/12/11 Python
使用numpy和PIL进行简单的图像处理方法
2018/07/02 Python
Python 十六进制整数与ASCii编码字符串相互转换方法
2018/07/09 Python
Python实现寻找回文数字过程解析
2020/06/09 Python
python自动生成证件号的方法示例
2021/01/14 Python
H5 canvas中width、height和style的宽高区别详解
2018/11/02 HTML / CSS
HTML5 自动聚焦(autofocus)属性使用介绍
2013/08/07 HTML / CSS
行政助理岗位职责范文
2013/12/03 职场文书
《再别康桥》教学反思
2014/02/12 职场文书
党员公开承诺书和承诺事项
2014/03/25 职场文书
幼儿园小班家长寄语
2014/04/02 职场文书
公司晚会策划方案
2014/05/17 职场文书
推广普通话标语
2014/06/27 职场文书
十佳好少年事迹材料
2014/08/21 职场文书
依法行政工作汇报
2014/10/28 职场文书
神农溪导游词
2015/02/11 职场文书
司机个人年终总结
2015/03/03 职场文书
工程款申请报告
2015/05/15 职场文书
js中Object.create实例用法详解
2021/10/05 Javascript