迭代器和生成器的區(qū)別

對于list胜臊、string鞋诗、tuple筒主、dict等這些容器對象,使用for循環(huán)遍歷是很方便的燥翅。在后臺for語句對容器對象調(diào)用iter()函數(shù)骑篙。iter()是Python內(nèi)置函數(shù)。

iter()會返回一個定義了next()方法的迭代器對象森书,它在容器中逐個訪問容器內(nèi)的元素靶端。next()也是python內(nèi)置函數(shù)谎势。在沒有后續(xù)元素時,next()會拋出一個StopIteration異常杨名,通知for語句循環(huán)結(jié)束脏榆。

一些術(shù)語的解釋:

1,迭代器協(xié)議:對象需要提供next()方法台谍,它要么返回迭代中的下一項须喂,要么就引起一個StopIteration異常,以終止迭代趁蕊。

2坞生,可迭代對象:實現(xiàn)了迭代器協(xié)議對象。list掷伙、tuple是己、dict都是Iterable(可迭代對象),但不是iteration(迭代器對象)任柜。但可以使用內(nèi)建函數(shù)iter()卒废,把這些都變成iteration(迭代器對象)。

Python自帶容器對象案例:

# 隨便定義一個listlistArray=[1,2,3]# 使用iter()函數(shù)iterName=iter(listArray)print(iterName)# 結(jié)果如下:是一個列表list的迭代器# print(next(iterName))print(next(iterName))print(next(iterName))print(next(iterName))#沒有迭代到下一個元素乘盼,直接拋出異常# 1# 2# 3# Traceback (most recent call last):#? File "Test07.py", line 32, in # StopIteration

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

Python中自定義類需要遍歷的案例:

# 計算出菲波那切數(shù)列classFib(object):def__init__(self, max):super(Fib, self).__init__()? ? ? ? self.max = maxdef__iter__(self):self.a =0self.b =1returnselfdef__next__(self):fib = self.aiffib > self.max:raiseStopIteration? ? ? ? self.a, self.b = self.b, self.a + self.breturnfib# 定義一個main函數(shù)升熊,循環(huán)遍歷每一個菲波那切數(shù)defmain():# 20以內(nèi)的數(shù)fib = Fib(20)foriinfib:? ? ? ? print(i)# 測試if__name__ =='__main__':? ? main()

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

解釋說明:

在本類的實現(xiàn)中,定義了一個iter(self)方法绸栅,這個方法是在for循環(huán)遍歷時被iter()調(diào)用级野,返回一個迭代器。因為在遍歷的時候粹胯,是直接調(diào)用的python內(nèi)置函數(shù)iter()蓖柔,由iter()通過調(diào)用iter(self)獲得對象的迭代器。有了迭代器风纠,就可以逐個遍歷元素了况鸣。而逐個遍歷的時候,也是使用內(nèi)置的next()函數(shù)通過調(diào)用對象的next(self)方法對迭代器對象進(jìn)行遍歷竹观。所以要實現(xiàn)iter(self)和next(self)镐捧。

而且因為實現(xiàn)了next(self),所以在實現(xiàn)iter(self)的時候臭增,直接返回self就可以懂酱。

總結(jié)一句話就是:在循環(huán)遍歷自定義容器對象時,會使用python內(nèi)置函數(shù)iter()調(diào)用遍歷對象的iter(self)獲得一個迭代器,之后再循環(huán)對這個迭代器使用next()調(diào)用迭代器對象的next(self)。

注意點:iter(self)只會被調(diào)用一次,而next(self)會被調(diào)用 n 次誊抛,直到出現(xiàn)StopIteration異常列牺。

Num02–>生成器

作用:

>延遲操作。也就是在需要的時候才產(chǎn)生結(jié)果拗窃,不是立即產(chǎn)生結(jié)果瞎领。

注意事項:

>生成器是只能遍歷一次的泌辫。

分類:

第一類:生成器函數(shù):還是使用? def? 定義函數(shù),但是九默,使用yield而不是return語句返回結(jié)果震放。yield語句一次返回一個結(jié)果,在每個結(jié)果中間荤西,掛起函數(shù)的狀態(tài)澜搅,以便下次從它離開的地方繼續(xù)執(zhí)行伍俘。

如下案例加以說明:

# 菲波那切數(shù)列defFib(max):n, a, b =0,0,1whilen < max:yieldb? ? ? ? a, b = b, a + b? ? ? ? n = n +1return'親邪锌!沒有數(shù)據(jù)了...'# 調(diào)用方法,生成出10個數(shù)來f=Fib(10)# 使用一個循環(huán)捕獲最后return 返回的值癌瘾,保存在異常StopIteration的value中whileTrue:try:? ? ? ? x=next(f)? ? ? ? print("f:",x)exceptStopIterationase:? ? ? ? print("生成器最后的返回值是:",e.value)break

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

第二類:生成器表達(dá)式:類似于列表推導(dǎo)觅丰,只不過是把一對大括號[]變換為一對小括號()。但是妨退,生成器表達(dá)式是按需產(chǎn)生一個生成器結(jié)果對象妇萄,要想拿到每一個元素,就需要循環(huán)遍歷咬荷。

如下案例加以說明:

# 一個列表xiaoke=[2,3,4,5]# 生成器generator冠句,類似于list,但是是把[]改為()gen=(aforainxiaoke)foriingen:? ? print(i)#結(jié)果是:2345# 為什么要使用生成器幸乒?因為效率懦底。# 使用生成器表達(dá)式取代列表推導(dǎo)式可以同時節(jié)省 cpu 和 內(nèi)存(RAM)。# 如果你構(gòu)造一個列表(list)的目的僅僅是傳遞給別的函數(shù),# 比如 傳遞給tuple()或者set(), 那就用生成器表達(dá)式替代吧!# 本案例是直接把列表轉(zhuǎn)化為元組kk=tuple(aforainxiaoke)print(kk)#結(jié)果是:(2,3,4,5)# python內(nèi)置的一些函數(shù)罕扎,可以識別這是生成器表達(dá)式result1=sum(aforainrange(3))print(result1)# 列表推導(dǎo)式result2=sum([aforainrange(3)])print(result2)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末聚唐,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子腔召,更是在濱河造成了極大的恐慌杆查,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,509評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件臀蛛,死亡現(xiàn)場離奇詭異亲桦,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)浊仆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評論 3 394
  • 文/潘曉璐 我一進(jìn)店門客峭,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人氧卧,你說我怎么就攤上這事桃笙。” “怎么了沙绝?”我有些...
    開封第一講書人閱讀 163,875評論 0 354
  • 文/不壞的土叔 我叫張陵搏明,是天一觀的道長鼠锈。 經(jīng)常有香客問我,道長星著,這世上最難降的妖魔是什么购笆? 我笑而不...
    開封第一講書人閱讀 58,441評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮虚循,結(jié)果婚禮上同欠,老公的妹妹穿的比我還像新娘。我一直安慰自己横缔,他們只是感情好铺遂,可當(dāng)我...
    茶點故事閱讀 67,488評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著茎刚,像睡著了一般襟锐。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上膛锭,一...
    開封第一講書人閱讀 51,365評論 1 302
  • 那天粮坞,我揣著相機(jī)與錄音,去河邊找鬼初狰。 笑死莫杈,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的奢入。 我是一名探鬼主播筝闹,決...
    沈念sama閱讀 40,190評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼俊马!你這毒婦竟也來了丁存?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,062評論 0 276
  • 序言:老撾萬榮一對情侶失蹤柴我,失蹤者是張志新(化名)和其女友劉穎解寝,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體艘儒,經(jīng)...
    沈念sama閱讀 45,500評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡聋伦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,706評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了界睁。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片觉增。...
    茶點故事閱讀 39,834評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖翻斟,靈堂內(nèi)的尸體忽然破棺而出逾礁,到底是詐尸還是另有隱情,我是刑警寧澤访惜,帶...
    沈念sama閱讀 35,559評論 5 345
  • 正文 年R本政府宣布嘹履,位于F島的核電站腻扇,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏砾嫉。R本人自食惡果不足惜幼苛,卻給世界環(huán)境...
    茶點故事閱讀 41,167評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望焕刮。 院中可真熱鬧舶沿,春花似錦、人聲如沸配并。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽荐绝。三九已至一汽,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間低滩,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評論 1 269
  • 我被黑心中介騙來泰國打工岩喷, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留恕沫,地道東北人。 一個月前我還...
    沈念sama閱讀 47,958評論 2 370
  • 正文 我出身青樓纱意,卻偏偏與公主長得像婶溯,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子偷霉,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,779評論 2 354

推薦閱讀更多精彩內(nèi)容