国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

理解Python的迭代器

2019-11-14 17:04:56
字體:
來源:轉載
供稿:網友

首先,廖雪峰老師的教程中解釋了迭代器和生成器,這篇文章只是補充和我個人的總結。

什么是迭代

可以直接作用于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]
  4. 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.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=-1    def __next__(self):        self.start +=2        return self.startI = 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.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 中測試通過。

日志:

  1. 8月13日完成
  2. 8月14日添加關于Iterator, Iterable的更多解釋在題外話的第4點。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 靖西县| 增城市| 军事| 信宜市| 兴仁县| 伊春市| 华坪县| 漳州市| 万年县| 德格县| 望城县| 潜江市| 临朐县| 兖州市| 蛟河市| 紫阳县| 阿坝| 同德县| 大洼县| 湾仔区| 泗水县| 来凤县| 股票| 阿合奇县| 平和县| 虹口区| 东明县| 江山市| 新安县| 灵寿县| 沧源| 洛川县| 南江县| 三亚市| 侯马市| 连平县| 陇南市| 桐庐县| 日土县| 西乌珠穆沁旗| 鹤壁市|