设为首页 收藏本站
查看: 704|回复: 0

[经验分享] 理解Python的迭代器

[复制链接]

尚未签到

发表于 2015-11-29 14:34:51 | 显示全部楼层 |阅读模式
  首先,廖雪峰老师的教程中解释了迭代器和生成器,这篇文章只是补充和我个人的总结。

什么是迭代
  可以直接作用于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(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(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点。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-144987-1-1.html 上篇帖子: 网站微信登录-python 实现 下篇帖子: linux下,Python 多版本共存,及Pip,Easy_install 安装扩展包
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表