自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Python關(guān)鍵字yield詳解

開(kāi)發(fā) 后端
此文由伯樂(lè)在線 –劉志軍編譯自stackoverflow Python標(biāo)簽中投票率最高的一個(gè)問(wèn)題《The Python yield keyword explained》,e-satis 詳細(xì)回答了關(guān)于yield 以及 generator、iterable、iterator、iteration之間的關(guān)系。

迭代器(Iterator)

為了理解yield是什么,首先要明白生成器(generator)是什么,在講生成器之前先說(shuō)說(shuō)迭代器(iterator),當(dāng)創(chuàng)建一個(gè)列表(list)時(shí),你可以逐個(gè)的讀取每一項(xiàng),這就叫做迭代(iteration)。

  1. mylist = [123]   
  2.  for i in mylist :   
  3.  print(i)   
  4. 1 
  5. 2 
  6. 3 

Mylist就是一個(gè)迭代器,不管是使用復(fù)雜的表達(dá)式列表,還是直接創(chuàng)建一個(gè)列表,都是可迭代的對(duì)象。

  1. mylist = [x*x for x in range(3)]   
  2. for i in mylist :   
  3. print(i)   
  4. 0 
  5. 1 
  6. 4 

你可以使用“for··· in ···”來(lái)操作可迭代對(duì)象,如:list,string,files,這些迭代對(duì)象非常方便我們使用,因?yàn)槟憧梢园凑漳愕囊庠高M(jìn)行重復(fù)的讀取。但是你不得不預(yù)先存儲(chǔ)所有的元素在內(nèi)存中,那些對(duì)象里有很多元素時(shí),并不是每一項(xiàng)都對(duì)你有用。

生成器(Generators)

生成器同樣是可迭代對(duì)象,但是你只能讀取一次,因?yàn)樗](méi)有把所有值存放內(nèi)存中,它動(dòng)態(tài)的生成值:

  1. mygenerator = (x*x for x in range(3))   
  2. for i in mygenerator :   
  3. print(i)   
  4. 0 
  5. 1 
  6. 4 

使用()和[]結(jié)果是一樣的,但是,第二次執(zhí)行“ for in mygenerator”不會(huì)有任何結(jié)果返回,因?yàn)樗荒苁褂靡淮?。首先?jì)算0,然后計(jì)算1,之后計(jì)算4,依次類推。

Yield

Yield是關(guān)鍵字, 用起來(lái)像return,yield在告訴程序,要求函數(shù)返回一個(gè)生成器。

  1. def createGenerator() :   
  2. mylist = range(3)   
  3. for i in mylist :   
  4. yield i*i   
  5.     
  6. mygenerator = createGenerator() # create a generator   
  7. print(mygenerator) # mygenerator is an object!   
  8. <generator object createGenerator at 0xb7555c34>   
  9. for i in mygenerator:   
  10. print(i)   
  11. 0 
  12. 1 
  13. 4 

這個(gè)示例本身沒(méi)什么意義,但是它很清晰地說(shuō)明函數(shù)將返回一組僅能讀一次的值,要想掌握yield,首先必須理解的是:當(dāng)你調(diào)用生成器函數(shù)的時(shí)候,如上例中的createGenerator(),程序并不會(huì)執(zhí)行函數(shù)體內(nèi)的代碼,它僅僅只是返回生成器對(duì)象,這種方式頗為微妙。函數(shù)體內(nèi)的代碼只有直到每次循環(huán)迭代(for)生成器的時(shí)候才會(huì)運(yùn)行。

函數(shù)***次運(yùn)行時(shí),它會(huì)從函數(shù)開(kāi)始處直到碰到y(tǒng)ield時(shí),就返回循環(huán)的***個(gè)值,然后,交互的運(yùn)行、返回,直到?jīng)]有值返回為止。如果函數(shù)在運(yùn)行但是并沒(méi)有遇到y(tǒng)ield,就認(rèn)為該生成器是空,原因可能是循環(huán)終止,或者沒(méi)有滿足任何”if/else”。

接下來(lái)讀一小段代碼來(lái)理解生成器的優(yōu)點(diǎn):

控制生成器窮舉

  1. >>> class Bank(): # 創(chuàng)建銀行,構(gòu)造ATM機(jī)   
  2. ...    crisis = False 
  3. ...    def create_atm(self) :   
  4. ...        while not self.crisis :   
  5. ...            yield "$100" 
  6. >>> hsbc = Bank() # 沒(méi)有危機(jī)時(shí),你想要多少,ATM就可以吐多少   
  7. >>> corner_street_atm = hsbc.create_atm()   
  8. >>> print(corner_street_atm.next())   
  9. $100 
  10. >>> print(corner_street_atm.next())   
  11. $100 
  12. >>> print([corner_street_atm.next() for cash in range(5)])   
  13. ['$100''$100''$100''$100''$100']   
  14. >>> hsbc.crisis = True # 危機(jī)來(lái)臨,銀行沒(méi)錢(qián)了   
  15. >>> print(corner_street_atm.next())   
  16. <type 'exceptions.StopIteration'>   
  17. >>> wall_street_atm = hsbc.ceate_atm() # 新建ATM,銀行仍然沒(méi)錢(qián)   
  18. >>> print(wall_street_atm.next())   
  19. <type 'exceptions.StopIteration'>   
  20. >>> hsbc.crisis = False # 麻煩就是,即使危機(jī)過(guò)后銀行還是空的   
  21. >>> print(corner_street_atm.next())   
  22. <type 'exceptions.StopIteration'>   
  23. >>> brand_new_atm = hsbc.create_atm() # 構(gòu)造新的ATM,恢復(fù)業(yè)務(wù)   
  24. >>> for cash in brand_new_atm :   
  25. ...    print cash   
  26. $100 
  27. $100 
  28. $100 
  29. $100 
  30. $100 
  31. $100 
  32. $100 
  33. $100 
  34. $100 

對(duì)于訪問(wèn)控制資源,生成器顯得非常有用。

迭代工具,你***的朋友

迭代工具模塊包含了操做指定的函數(shù)用于操作迭代器。想復(fù)制一個(gè)迭代器出來(lái)?鏈接兩個(gè)迭代器?以one liner(這里的one-liner只需一行代碼能搞定的任務(wù))用內(nèi)嵌的列表組合一組值?不使用list創(chuàng)建Map/Zip?···,你要做的就是 import itertools,舉個(gè)例子吧:

四匹馬賽跑到達(dá)終點(diǎn)排名的所有可能性:

  1. >>> horses = [1234]   
  2. >>> races = itertools.permutations(horses)   
  3. >>> print(races)   
  4. <itertools.permutations object at 0xb754f1dc>   
  5. >>> print(list(itertools.permutations(horses)))   
  6. [(1234),   
  7.  (1243),   
  8.  (1324),   
  9.  (1342),   
  10.  (1423),   
  11.  (1432),   
  12.  (2134),   
  13.  (2143),   
  14.  (2314),   
  15.  (2341),   
  16.  (2413),   
  17.  (2431),   
  18.  (3124),   
  19.  (3142),   
  20.  (3214),   
  21.  (3241),   
  22.  (3412),   
  23.  (3421),   
  24.  (4123),   
  25.  (4132),   
  26.  (4213),   
  27.  (4231),   
  28.  (4312),   
  29.  (4321)] 

理解迭代的內(nèi)部機(jī)制:

迭代(iteration)就是對(duì)可迭代對(duì)象(iterables,實(shí)現(xiàn)了__iter__()方法)和迭代器(iterators,實(shí)現(xiàn)了__next__()方法)的一個(gè)操作過(guò)程??傻鷮?duì)象是任何可返回一個(gè)迭代器的對(duì)象,迭代器是應(yīng)用在迭代對(duì)象中迭代的對(duì)象,換一種方式說(shuō)的話就是:iterable對(duì)象的__iter__()方法可以返回iterator對(duì)象,iterator通過(guò)調(diào)用next()方法獲取其中的每一個(gè)值(譯者注),讀者可以結(jié)合Java API中的 Iterable接口和Iterator接口進(jìn)行類比。

英文原文:The Python yield keyword explained

原文鏈接:http://blog.jobbole.com/32748/

責(zé)任編輯:張偉 來(lái)源: 伯樂(lè)在線
相關(guān)推薦

2009-12-18 11:37:54

Ruby關(guān)鍵字yiel

2023-12-11 13:59:00

YieldPython生成器函數(shù)

2009-09-02 09:24:03

C# this關(guān)鍵字

2019-08-29 09:11:38

Pythonyield語(yǔ)法

2011-06-14 13:26:27

volatile

2022-11-12 18:32:50

Golangomitemptyjson

2021-02-01 13:10:07

Staticc語(yǔ)言UNIX系統(tǒng)

2025-01-22 08:06:38

C#yield數(shù)據(jù)迭代

2018-04-20 15:56:09

Pythonglobal關(guān)鍵字

2021-07-27 07:31:16

單例模式關(guān)鍵字

2009-09-28 11:34:49

Javascript

2023-03-09 07:38:58

static關(guān)鍵字狀態(tài)

2010-02-05 15:51:06

C++ explici

2017-05-27 20:59:30

Java多線程synchronize

2023-11-10 09:29:30

MySQLExplain

2024-03-15 11:52:03

C++關(guān)鍵字編程

2022-05-06 08:32:40

Pythonwith代碼

2009-08-26 09:58:22

C#關(guān)鍵字

2020-08-23 11:03:24

Python開(kāi)發(fā)void

2009-08-21 14:58:56

C# this關(guān)鍵字
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)