摘要:簡評迭代器是惰性可迭代對象,函數(shù)在中是一個惰性的可迭代對象,那么是不是迭代器呢為什么。如果你不能將某些東西傳遞給函數(shù),那么它不是一個迭代器。的對象不是迭代器。
簡評:迭代器(iterator)是惰性可迭代對象(lazy iterable),range 函數(shù)在 Python 3 中是一個惰性的可迭代對象,那么 range 是不是迭代器呢?為什么。
TLNR:Python 3 中的 range 對象(Python 2 中的 xrange 對象)是 lazy 的,但 range 對象卻不是迭代器。
是的,這讓人很困惑
當(dāng)談?wù)?Python 中的迭代器(iterator)和可迭代對象(iterable)時,你很可能會聽到有人重復(fù) range 是迭代器的誤解。我認(rèn)為這是非常嚴(yán)重誤解, 如果你認(rèn)為 range 對象是迭代器,那么你關(guān)于「迭代器是如何運(yùn)行」的心智模型還不夠清楚。從某種意義上來說,range 和迭代器都是「惰性」的,但它們是以相當(dāng)不同的方式實(shí)現(xiàn)「惰性」的。
什么是迭代器(iterator)
在 Python 中,可迭代對象就是你可以迭代的任何東西,而迭代器就是實(shí)際迭代的東西。
Iter-ables are able to be iterated over. Iter-ators are the agents that perform the iteration.
可以使用 iter 函數(shù)從任何可迭代對象中獲取迭代器:
一旦有了迭代器,可以用它做的唯一的事情就是獲得它的下一個元素:
如果沒有更多的元素了, 則會拋出一個 stop iteration exception:
所有的迭代器都是可迭代對象,意思是你可以從一個迭代器中得到一個迭代器,因此你可以遍歷一個迭代器:
應(yīng)該指出的是迭代器是有狀態(tài)的,在循環(huán)遍歷一次迭代器后,如果嘗試再次循環(huán),它將為空:
在 Python 3 中,enumerate、zip、reversed和其他一些內(nèi)置函數(shù)會返回迭代器:
生成器(無論來自生成器函數(shù)還是生成器表達(dá)式)是一種創(chuàng)建迭代器的簡單方法:
我經(jīng)常說迭代器是惰性的一次性可迭代對象。 「惰性」是因?yàn)樗麄冎谎h(huán)計(jì)算項(xiàng)目,「單次使用是因?yàn)橐坏囊粋€迭代器中「消費(fèi)」了一個元素之后,這個元素就永遠(yuǎn)消失了。
什么是 range
Python 3 中的 range 對象(Python 2 中的 xrange)可以像任何其他可迭代對象一樣循環(huán)使用:
因?yàn)?range 是可迭代對象,所以可以從中得到一個迭代器:
但 range 對象本身不是迭代器,我們不能在 range 對象上調(diào)用 next:
與迭代器不同的是,我們可以遍歷一個 range 對象而不「消耗」它:
如果我們使用迭代器完成此操作,則第二次循環(huán)時不會得到任何元素:
宗上,與 zip, enumerate, or generator對象不同,range 對象不是迭代器。
那么,究竟 range 是什么
range 對象在某種意義上是「惰性的」,因?yàn)樗粫蓜?chuàng)建時包含的每個數(shù)字,相反,當(dāng)我們在循環(huán)中需要的時候,它才將這些數(shù)字返回給我們。
下面是一個 range 對象和一個生成器(是一種迭代器):
不像生成器,range 對象有長度:
并且可以被索引:
與迭代器不同,你可以詢問他們是否包含某元素而不改變他們的狀態(tài):
如果你想要一個 range 對象的描述,可以稱它們?yōu)閼行蛄?,range 是序列(如列表,元組和字符串),但并不包含任何內(nèi)存中的內(nèi)容,而是通過計(jì)算來回答問題。
為什么這個區(qū)別很重要
如果我告訴你某個對象是一個迭代器,你會知道當(dāng)在這個對象上調(diào)用 iter 函數(shù)時,總會得到相同的的對象(按照定義):
確信可以在這個對象上調(diào)用 next 函數(shù),因?yàn)榭梢栽谒械牡魃险{(diào)用 next 函數(shù):
而且你會知道,當(dāng)遍歷它時,這些元素將從迭代器中被消耗掉,有時候這個特性可以派上用場(以特殊的方式處理迭代器):
所以雖然看起來「惰性可迭代對象」和「迭代器」之間的區(qū)別很微妙,但這些術(shù)語確實(shí)意味著不同的東西。 雖然「惰性可迭代對象」是一個沒有具體含義的非常普遍的術(shù)語,但「迭代器」這個詞意味著一個具有非常特定行為的對象。
總結(jié)
如果你知道你可以循環(huán)遍歷某個對象,這是一個可迭代對象(iterable)。
如果你知道你正在循環(huán)遍歷的對象是在循環(huán)的時候計(jì)算出來,那么這是一個惰性可迭代對象(lazy iterable)。
如果你知道你可以傳遞一些東西給 next 函數(shù),它就是一個迭代器(這是最常見的惰性可迭代對象)。
如果你可以循環(huán)多次而不用「耗盡」它,它不是一個迭代器。如果你不能將某些東西傳遞給 next 函數(shù),那么它不是一個迭代器。 Python 3 的 range 對象不是迭代器。 如果你正在指導(dǎo)別人關(guān)于 range 對象的知識,請不要使用「迭代器」一詞,這會讓人十分困惑,并可能導(dǎo)致他人開始濫用「迭代器」這個詞。
原文:Python: range is not an iterator!
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/44627.html
摘要:為什么不更規(guī)范點(diǎn),令生成的是迭代器呢關(guān)于這個問題,我沒找到官方解釋,以下純屬個人觀點(diǎn)。類型是什么以上是我對為什么不產(chǎn)生迭代器的一種解答。小結(jié)回顧全文,我得到了兩個偏冷門的結(jié)論是可迭代對象而不是迭代器對象是不可變的等差序列。 showImg(https://segmentfault.com/img/bVbmKjd?w=6000&h=4000); 迭代器是 23 種設(shè)計(jì)模式中最常用的一種(...
摘要:上一篇文章標(biāo)準(zhǔn)庫內(nèi)置類型數(shù)字類型下一篇文章標(biāo)準(zhǔn)庫內(nèi)置類型文本序列類型迭代器類型支持在容器中進(jìn)行迭代的概念。該對象需要支持下文所述的迭代器協(xié)議。這是同時允許容器和迭代器配合和語句使用所必須的。 上一篇文章:Python標(biāo)準(zhǔn)庫---10、內(nèi)置類型:數(shù)字類型下一篇文章:Python標(biāo)準(zhǔn)庫---12、內(nèi)置類型:文本序列類型(str) ## 迭代器類型Python 支持在容器中進(jìn)行迭代的概念。...
閱讀 823·2023-04-25 20:18
閱讀 2111·2021-11-22 13:54
閱讀 2551·2021-09-26 09:55
閱讀 3916·2021-09-22 15:28
閱讀 2990·2021-09-03 10:34
閱讀 1726·2021-07-28 00:15
閱讀 1651·2019-08-30 14:25
閱讀 1292·2019-08-29 17:16