转载自理解Python的迭代器

首先,廖雪峰老师的教程中解释了迭代器和生成器,这篇文章只是补充和我个人的总结。

什么是迭代
可以直接作用于for循环的对象统称为可迭代对象(Iterable)。
可以被next()函数调用并不断返回下一个值的对象称为迭代器(Iterator)。
所有的Iterable均可以通过内置函数iter()来转变为Iterator。

对迭代器来讲,有一个next()就够了。在你使用for 和 in 语句时,程序就会自动调用即将被处理的对象的迭代器对象,然后使用它的next()方法,直到监测到一个StopIteration异常。

>>> L = [1,2,3]
>>> [x**2 for x in L]
[1, 4, 9]
>>> next(L)
Traceback (most recent call last):File "<stdin>", line 1, in <module>
TypeError: 'list' object is not an iterator
>>> I=iter(L)
>>> next(I)
1
>>> next(I)
2
>>> next(I)
3
>>> next(I)
Traceback (most recent call last):File "<stdin>", line 1, in <module>
StopIteration

上面例子中,列表L可以被for进行循环但是不能被内置函数next()用来查找下一个值,所以L是Iterable。
L通过iter进行包装后设为I,I可以被next()用来查找下一个值,所以I是Iterator。

题外话:

  1. 内置函数iter()仅仅是调用了对象的iter()方法,所以list对象内部一定存在方法iter()
  2. 内置函数next()仅仅是调用了对象的next()方法,所以list对象内部一定不存在方法next(),但是Itrator中一定存在这个方法。

  3. for循环内部事实上就是先调用iter()把Iterable变成Iterator在进行循环迭代的。

>>> L = [4,5,6]
>>> I = L.__iter__()
>>> L.__next__()
Traceback (most recent call last):File "<stdin>", line 1, in <module>
AttributeError: 'list' object has no attribute '__next__'
>>> I.__next__()
4
>>> from collections import Iterator, Iterable
>>> isinstance(L, Iterable)
True
>>> isinstance(L, Iterator)
False
>>> isinstance(I, Iterable)
True
>>> isinstance(I, Iterator)
True
>>> [x**2 for x in I]
[25, 36]
  1. Iterator继承自Iterable,从下面的测试中可以很方便的看到Iterator包含iter()和next()方法,而Iteratble仅仅包含iter()。
>>> from collections import Iterator, Iterable
>>> help(Iterator)
Help on class Iterator:class Iterator(Iterable)|  Method resolution order:|      Iterator|      Iterable|      builtins.object   |**注解:从这里可以看出Iterable继承自object, Iterator继承自Iterable。|  Methods defined here:||  __iter__(self)||  __next__(self)|      Return the next item from the iterator. When exhausted, raise StopIteration
......
>>> help(Iterable)
Help on class Iterable:class Iterable(builtins.object)|  Methods defined here:||  __iter__(self)
......

iterable需要包含有_iter()方法用来返回iterator,而iterator需要包含有next_()方法用来被循环
如果我们自己定义迭代器,只要在类里面定义一个 iter() 函数,用它来返回一个带 next() 方法的对象就够了。
直接上代码

class Iterable:
def iter(self):
return Iterator()

class Iterator:def __init__(self):self.start=-1def __next__(self):self.start +=2if self.start >10:raise StopIterationreturn self.startI = Iterable()
for i in I:print(i)

上面的代码实现的是找到10以内的奇数,代码中的类名可以随便取,不是一定需要使用我上面提供的类名的。
如果在Iterator的next方法中没有实现StopIteration异常,那么则是表示的全部奇数,那么需要在调用的时候设置退出循环的条件。

class Iterable:def __iter__(self):return Iterator()class Iterator:def __init__(self):self.start=-1def __next__(self):self.start +=2return self.startI = Iterable()
for count, i in zip(range(5),I):    #也可以用内置函数enumerate来实现计数工作。print(i)

我们通过range来实现打印多少个元素,这里表示打印5个元素,返回结果和上面一致。

当然,我们可以把这两个类合并在一起,这样实现程序的简练。
最终版本如下

class Iterable:def __iter__(self):return selfdef __init__(self):self.start=-1def __next__(self):self.start +=2if self.start >10:raise StopIterationreturn self.startI = Iterable()
for i in I:print(i)

复制迭代器
迭代器是一次性消耗品,使用完了以后就空了,请看。

>>> L=[1,2,3]
>>> I=iter(L)
>>> for i in I:
...     print(i, end='-')
...
1-2-3-
>>>next(I)
Traceback (most recent call last):File "<stdin>", line 1, in <module>
StopIteration

当循环以后就殆尽了,再次使用调用时会引发StopIteration异常。

我们想通过直接赋值的形式把迭代器保存起来,可以下次使用。
但是通过下面的范例可以看出来,根本不管用。

>>> I=iter(L)
>>> J=I
>>> next(I)
1
>>> next(J)
2
>>> next(I)
3
>>> next(J)
Traceback (most recent call last):File "<stdin>", line 1, in <module>
StopIteration

那怎么样才能达到我们要的效果呢?
我们需要使用copy包中的deepcopy了,请看下面:

>>> import copy
>>> I=iter(L)
>>> J=copy.deepcopy(I)
>>> next(I)
1
>>> next(I)
2
>>> next(J)
1

补充:迭代器不能向后移动, 不能回到开始。
所以需要做一些特殊的事情才能实现向后移动等功能。

以上代码均在Python 3.4 中测试通过。

python学习笔记--迭代器相关推荐

  1. [转载] Python 学习笔记 迭代器和生成器

    参考链接: Python中的迭代器函数2(islice(),starmap(),tee()..) 本文链接地址 http://quqiuzhu.com/2016/python-iterator-and ...

  2. Python学习笔记三之编程练习:循环、迭代器与函数

    Python学习笔记三之编程练习 1. 编程第一步 # 求解斐波纳契数列 #/user/bin/python3#Fibonacci series:斐波那契数列 #两个元素的总和确定了下一个数 a,b= ...

  3. Python学习笔记(十一)

    Python学习笔记(十一): 生成器,迭代器回顾 模块 作业-计算器 1. 生成器,迭代器回顾 1. 列表生成式:[x for x in range(10)] 2. 生成器 (generator o ...

  4. python学习笔记目录

    人生苦短,我学python学习笔记目录: week1 python入门week2 python基础week3 python进阶week4 python模块week5 python高阶week6 数据结 ...

  5. python基本语法语句-python学习笔记:基本语法

    原标题:python学习笔记:基本语法 缩进:必须使用4个空格来表示每级缩进,支持Tab字符 if语句,经常与else, elif(相当于else if) 配合使用. for语句,迭代器,依次处理迭代 ...

  6. Python学习笔记:常用内建模块4:hmac

    前言 最近在学习深度学习,已经跑出了几个模型,但Pyhton的基础不够扎实,因此,开始补习Python了,大家都推荐廖雪峰的课程,因此,开始了学习,但光学有没有用,还要和大家讨论一下,因此,写下这些帖 ...

  7. Python学习笔记:生成器(Generator)

    Python学习笔记:生成器(Generator) 拥有迭代序列的一致方式,比如列表里的对象或文件里的行,是Python的一项重要特性.这是通过迭代器协议的方式来完成的,一种生成可迭代对象的通用方式. ...

  8. python 学习笔记 (核心)

    python    学习笔记 (核心) Python解释器从头到尾一行接一行执行脚本 # -*- coding: UTF-8 -*-    //字符编码 不区分单引号和双引号,x='hello',x[ ...

  9. Python学习笔记第二十九天(N维数组(ndarray))

    Python学习笔记第二十九天 N维数组(ndarray) 构建阵列 索引阵列 ndarray的内部内存布局 阵列属性 内存布局 数据类型 其他属性 阵列接口 ctypes外部功能接口 Array方法 ...

最新文章

  1. ajax实现简单计算器,一个简单的jQuery计算器实现了连续计算功能
  2. Cell重磅:记忆研究的突破进展!在诺奖成果基础上,用“全光学”组合来“操纵记忆”...
  3. Javascript之DOM(Document类型)
  4. 天天沉迷于皇上本宫的都是sb
  5. 打造vim的python编辑器
  6. mysql8支持myISAM_mysql8 参考手册--优化MyISAM表
  7. Java Currency getInstance()方法与示例
  8. DOM701:禁用了反向和正向缓存(开发者)
  9. 网页设计中的中国传统色彩速查表 颜色值
  10. python中flag=1什么意思_001_flag包详解
  11. 缓解疲劳、舒缓全身放松,游养乐分享养生小秘籍
  12. apriori java_频繁模式挖掘apriori算法介绍及Java实现
  13. 记一次windowns7系统IE闪退恢复解决过程
  14. vsftp,lftp
  15. 聊聊微软的两个产品XBOX和SharePoint
  16. cisco 3560 QOS配置
  17. Python+AI让静态图片动起来
  18. 关于mysql的国际化
  19. 企业OA系统选型误区:OA就是传统的公文管理和综合办公工具
  20. HDU - 4489 The King’s Ups and Downs(dp)

热门文章

  1. java 权限控制 demo_Java-访问控制权限
  2. 【51单片机快速入门指南】5.1:SPI与DS1302时钟芯片
  3. Hi3520D UART2和UART3是如何加载到内核的
  4. h264检测是I帧还是P帧
  5. 接触VC之四:COM组件模型基础
  6. [react-router] 请你说说react的路由是什么?
  7. react学习(68)--ant design inputNumber
  8. [html] 如何优化大数据列表(10万+)的性能?说说你的方案
  9. 工作394-注册页面学习
  10. [html] 如何阻止屏幕旋转时自动调整字体的大小?