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

什么是迭代

可以直接作用于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 "", line 1, in

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 "", line 1, in

StopIteration

上面例子中,列表L可以被for进行循环但是不能被内置函数next()用来查找下一个值,所以L是Iterable。

L通过iter进行包装后设为I,I可以被next()用来查找下一个值,所以I是Iterator。

题外话:

内置函数iter()仅仅是调用了对象的__iter__()方法,所以list对象内部一定存在方法__iter__()

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

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

>>> L = [4,5,6]

>>> I = L.__iter__()

>>> L.__next__()

Traceback (most recent call last):

File "", line 1, in

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]

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=-1

def __next__(self):

self.start +=2

if self.start >10:

raise StopIteration

return self.start

I = 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=-1

def __next__(self):

self.start +=2

return self.start

I = Iterable()

for count, i in zip(range(5),I): #也可以用内置函数enumerate来实现计数工作。

print(i)

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

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

最终版本如下

class Iterable:

def __iter__(self):

return self

def __init__(self):

self.start=-1

def __next__(self):

self.start +=2

if self.start >10:

raise StopIteration

return self.start

I = 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 "", line 1, in

StopIteration

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

我们想通过直接赋值的形式把迭代器保存起来,可以下次使用。

但是通过下面的范例可以看出来,根本不管用。

>>> I=iter(L)

>>> J=I

>>> next(I)

1

>>> next(J)

2

>>> next(I)

3

>>> next(J)

Traceback (most recent call last):

File "", line 1, in

StopIteration

那怎么样才能达到我们要的效果呢?

我们需要使用copy包中的deepcopy了,请看下面:

>>> import copy

>>> I=iter(L)

>>> J=copy.deepcopy(I)

>>> next(I)

1

>>> next(I)

2

>>> next(J)

1

补充:迭代器不能向后移动, 不能回到开始。

所以需要做一些特殊的事情才能实现向后移动等功能。

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

日志:

8月13日完成

8月14日添加关于Iterator, Iterable的更多解释在题外话的第4点。

python迭代器是什么意思_理解Python的迭代器相关推荐

  1. python中get啥意思_理解Python中的.get()方法

    本问题已经有最佳答案,请猛点这里访问. sentence ="The quick brown fox jumped over the lazy dog." characters = ...

  2. python中的元类_理解python中的元类

    一,理解类也是对象 在python中类同样也是一种对象,只要使用关键字class,Python解释器在执行的时候就会创建一个对象,这个对象(类)自身拥有创建对象(类实例)的能力,这就是为什么他是一个类 ...

  3. python正则表达式是什么意思_理解python正则表达式

    在python中,对正则表达式的支持是通过re模块来支持的.使用re的步骤是先把表达式字符串编译成pattern实例,然后在使用pattern去匹配文本获取结果. 其实也有另外一种方式,就是直接使用r ...

  4. python中加入绝对路径_理解Python中的绝对路径和相对路径

    本文介绍了Python中的绝对路劲和相对路径,分享给大家,也给自己留个笔记 1.绝对路径 os.path.abspath("文件名"): 显示的是一个文件的绝对路劲 eg: > ...

  5. python无返回值函数_理解Python 中无返回值函数的问题

    例如 list 的 append 操作就是无返回值的,换句话说就是不能进行形如 list = [] list.append(1).append(2) 这样的连续操作 注意函数返回的数据类型注意是 li ...

  6. c++ 协程_理解Python协程(Coroutine)

    由于GIL的存在,导致Python多线程性能甚至比单线程更糟. GIL: 全局解释器锁(英语:Global Interpreter Lock,缩写GIL),是计算机程序设计语言解释器用于同步线程的一种 ...

  7. python3 协程 写法_理解Python的协程(Coroutine)

    由于GIL的存在,导致Python多线程性能甚至比单线程更糟. GIL: 全局解释器锁(英语:Global Interpreter Lock,缩写GIL),是计算机程序设计语言解释器用于同步线程的一种 ...

  8. python双重for循环怎么理解_理解 Python 的 for 循环

    Python部落(python.freelycode.com)组织翻译,禁止转载,欢迎转发. 在本篇博客中,我们将讨论 Python 中 for 循环的原理. 我们将从一组基本例子和它的语法开始,还将 ...

  9. python中元组_理解python中的元组

    理解 python 中的元组 引言 在 Python 中元组是这样的: 元组是是这样一种数据结构:不变的或者不可改变的(简单来说不能重新赋值) .元素的有序序列.因为元组是 不变的,所以他的数值是不能 ...

最新文章

  1. 《PHP精粹:编写高效PHP代码》——第1章面向对象编程
  2. JAVA基础_修饰符
  3. Python3之Django框架搭建详细步骤
  4. git 本地 更新到 仓库 仅仅 操作系统课程设计
  5. 【SPSS】软件介绍
  6. CodeForces 650A Watchmen
  7. XP经典壁纸,多少人曾爱慕你年轻时的容颜
  8. 优秀的长截图标注工具:iShot for Mac
  9. win10 linux 无法下载,大神为你win10系统无法安装ubuntu的处理
  10. Unicode 中文,日文,西欧语言Unicode编码域(区间)
  11. 计算机专业实训图片,实训一图片的简单处理_计算机软件及应用_IT计算机_专业资料...
  12. android音乐搜索功能实现,撸个应用学Android——空灵音乐本地音乐版
  13. 记一次在学院服务器装Ubuntu系统
  14. 应用分发平台之苹果超级签名流程分析及API错误
  15. win10家庭版解决“管理员已阻止你运行此应用”
  16. 宁波跨境海关商品订单推送清关及支付宝海关报关
  17. top和margin-top等的区别
  18. 【亲测有效】解决执行nrm ls命令查看不到*星号的问题
  19. Aliyun短信服务集成
  20. 小学计算机兴趣班要教什么,小学以后,一般给孩子报几个兴趣班合适?

热门文章

  1. 用自定义函数联合IF函数实现“一对多”查询
  2. 完了!Python开挂! 90%的程序员:痛快!你怎么看? ​
  3. 任正非解读华为“狼文化”;丰巢高管:不会放弃超时收费;Debian 10.4 发布 | 极客头条...
  4. 如何检查手机上的 App 是不是正版?
  5. 腾讯面试官这样问我二叉树,我刚好都会 | 原力计划
  6. 罗永浩宣布进军电商直播;微博回应用户数据泄露;Android 11 开发者预览版 2 发布 | 极客头条...
  7. 2020 年区块链最具影响力人物 Top 20,孙宇晨也上榜了
  8. 钉钉辟谣“老师能打开学生摄像头”;HTC 关闭官方社区;​Node.js 安全版本发布 | 极客头条...
  9. 软件设计师的成长之路
  10. 如何用 Go 快速编写出 HTTP REST API 服务?