python yield的用法和詳解

之前一直對python的yield理解的有點(diǎn)模糊峻村,昨天搜索發(fā)現(xiàn)了一個(gè)講解非常清楚的版本锡凝,所以mark一下。
來源:python中yield的用法詳解——最簡單窜锯,最清晰的解釋

首先,如果你還沒有對yield有個(gè)初步分認(rèn)識吞瞪,那么你先把yield看做“return”芍秆,這個(gè)是直觀的,它首先是個(gè)return霉颠,普通的return是什么意思荆虱,就是在程序中返回某個(gè)值,返回之后程序就不再往下運(yùn)行了克伊。看做return之后再把它看做一個(gè)是生成器(generator)的一部分(帶yield的函數(shù)才是真正的迭代器)不从,好了犁跪,如果你對這些不明白的話,那先把yield看做return,然后直接看下面的程序坷衍,你就會(huì)明白yield的全部意思了:

def foo():
    print("starting...")
    while True:
        res = yield 4
        print("res:",res)
g = foo()
print(next(g))
print("*"*20)
print(next(g))

就這么簡單的幾行代碼就讓你明白什么是yield,代碼的輸出這個(gè):

starting...
4
********************
res: None
4

我直接解釋代碼運(yùn)行順序乏矾,相當(dāng)于代碼單步調(diào)試:

  1. 程序開始執(zhí)行以后迁杨,因?yàn)閒oo函數(shù)中有yield關(guān)鍵字,所以foo函數(shù)并不會(huì)真的執(zhí)行铅协,而是先得到一個(gè)生成器g(相當(dāng)于一個(gè)對象)

  2. 直到調(diào)用next方法,foo函數(shù)正式開始執(zhí)行痒给,先執(zhí)行foo函數(shù)中的print方法骏全,然后進(jìn)入while循環(huán)

  3. 程序遇到y(tǒng)ield關(guān)鍵字,然后把yield想想成return,return了一個(gè)4之后序仙,程序停止鲁豪,并沒有執(zhí)行賦值給res操作律秃,此時(shí)next(g)語句執(zhí)行完成治唤,所以輸出的前兩行(第一個(gè)是while上面的print的結(jié)果,第二個(gè)是return出的結(jié)果)是執(zhí)行print(next(g))的結(jié)果,

  4. 程序執(zhí)行print(""20)宾添,輸出20個(gè)*

  5. 又開始執(zhí)行下面的print(next(g)),這個(gè)時(shí)候和上面那個(gè)差不多,不過不同的是粱锐,這個(gè)時(shí)候是從剛才那個(gè)next程序停止的地方開始執(zhí)行的扛邑,也就是要執(zhí)行res的賦值操作,這時(shí)候要注意蔬崩,這個(gè)時(shí)候賦值操作的右邊是沒有值的(因?yàn)閯偛拍莻€(gè)是return出去了,并沒有給賦值操作的左邊傳參數(shù))跨琳,所以這個(gè)時(shí)候res賦值是None,所以接著下面的輸出就是res:None,

  6. 程序會(huì)繼續(xù)在while里執(zhí)行桐罕,又一次碰到y(tǒng)ield,這個(gè)時(shí)候同樣return 出4,然后程序停止侠鳄,print函數(shù)輸出的4就是這次return出的4.

到這里你可能就明白yield和return的關(guān)系和區(qū)別了死宣,帶yield的函數(shù)是一個(gè)生成器碴开,而不是一個(gè)函數(shù)了,這個(gè)生成器有一個(gè)函數(shù)就是next函數(shù)潦牛,next就相當(dāng)于“下一步”生成哪個(gè)數(shù),這一次的next開始的地方是接著上一次的next停止的地方執(zhí)行的朴爬,所以調(diào)用next的時(shí)候橡淆,生成器并不會(huì)從foo函數(shù)的開始執(zhí)行母赵,只是接著上一步停止的地方開始具滴,然后遇到y(tǒng)ield后,return出要生成的數(shù)构韵,此步就結(jié)束。


def foo():
    print("starting...")
    while True:
        res = yield 4
        print("res:",res)
g = foo()
print(next(g))
print("*"*20)
print(g.send(7))

再看一個(gè)這個(gè)生成器的send函數(shù)的例子凶朗,這個(gè)例子就把上面那個(gè)例子的最后一行換掉了显拳,輸出結(jié)果:

starting...
4
********************
res: 7
4

先大致說一下send函數(shù)的概念:此時(shí)你應(yīng)該注意到上面那個(gè)的紫色的字,還有上面那個(gè)res的值為什么是None遇八,這個(gè)變成了7耍休,到底為什么,這是因?yàn)檠蚓瑂end是發(fā)送一個(gè)參數(shù)給res的,因?yàn)樯厦嬷v到读规,return的時(shí)候燃少,并沒有把4賦值給res,下次執(zhí)行的時(shí)候只好繼續(xù)執(zhí)行賦值操作阵具,只好賦值為None了,而如果用send的話怕敬,開始執(zhí)行的時(shí)候帘皿,先接著上一次(return 4之后)執(zhí)行,先把7賦值給了res,然后執(zhí)行next的作用虽填,遇見下一回的yield,return出結(jié)果后結(jié)束卤唉。

5.程序執(zhí)行g(shù).send(7),程序會(huì)從yield關(guān)鍵字那一行繼續(xù)向下運(yùn)行竭恬,send會(huì)把7這個(gè)值賦值給res變量

6.由于send方法中包含next()方法熬的,所以程序會(huì)繼續(xù)向下運(yùn)行執(zhí)行print方法,然后再次進(jìn)入while循環(huán)

7.程序執(zhí)行再次遇到y(tǒng)ield關(guān)鍵字押框,yield會(huì)返回后面的值后,程序再次暫停盒揉,直到再次調(diào)用next方法或send方法兑徘。

這就結(jié)束了,說一下藕漱,為什么用這個(gè)生成器崭闲,是因?yàn)槿绻肔ist的話,會(huì)占用更大的空間刁俭,比如說取0,1,2,3,4,5,6............1000

你可能會(huì)這樣:

for n in range(1000):
   a=n

這個(gè)時(shí)候range(1000)就默認(rèn)生成一個(gè)含有1000個(gè)數(shù)的list了,所以很占內(nèi)存沙兰。

這個(gè)時(shí)候你可以用剛才的yield組合成生成器進(jìn)行實(shí)現(xiàn)翘魄,也可以用xrange(1000)這個(gè)生成器實(shí)現(xiàn)

yield組合:

def foo(num):
    print("starting...")
    while num<10:
        num=num+1
        yield num
for n in foo(0):
    print(n)

輸出:

starting...
1
2
3
4
5
6
7
8
9
10
xrange(1000):

for n in xrange(1000):
   a=n

其中要注意的是python3時(shí)已經(jīng)沒有xrange()了舀奶,在python3中,range()就是xrange()了但荤,你可以在python3中查看range()的類型,它已經(jīng)是個(gè)<class 'range'>了腹躁,而不是一個(gè)list了,畢竟這個(gè)是需要優(yōu)化的哑了。

作者:馮爽朗
來源:CSDN
原文:https://blog.csdn.net/mieleizhi0522/article/details/82142856

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末烧颖,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子拆火,更是在濱河造成了極大的恐慌涂圆,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,454評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件模狭,死亡現(xiàn)場離奇詭異卡辰,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)反砌,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評論 3 385
  • 文/潘曉璐 我一進(jìn)店門萌朱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人酒贬,你說我怎么就攤上這事翠霍。” “怎么了寒匙?”我有些...
    開封第一講書人閱讀 157,921評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長考蕾。 經(jīng)常有香客問我,道長蚯窥,這世上最難降的妖魔是什么塞帐? 我笑而不...
    開封第一講書人閱讀 56,648評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮矛紫,結(jié)果婚禮上牌里,老公的妹妹穿的比我還像新娘。我一直安慰自己牡辽,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,770評論 6 386
  • 文/花漫 我一把揭開白布麸澜。 她就那樣靜靜地躺著奏黑,像睡著了一般。 火紅的嫁衣襯著肌膚如雪馁害。 梳的紋絲不亂的頭發(fā)上蹂匹,一...
    開封第一講書人閱讀 49,950評論 1 291
  • 那天,我揣著相機(jī)與錄音忍啸,去河邊找鬼履植。 笑死,一個(gè)胖子當(dāng)著我的面吹牛玫霎,可吹牛的內(nèi)容都是我干的传泊。 我是一名探鬼主播鸭巴,決...
    沈念sama閱讀 39,090評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼鹃祖,長吁一口氣:“原來是場噩夢啊……” “哼普舆!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起沼侣,我...
    開封第一講書人閱讀 37,817評論 0 268
  • 序言:老撾萬榮一對情侶失蹤蛾洛,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后轧膘,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,275評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡鳞滨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,592評論 2 327
  • 正文 我和宋清朗相戀三年拯啦,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了熔任。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,724評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡碱蒙,死狀恐怖夯巷,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情趁餐,我是刑警寧澤,帶...
    沈念sama閱讀 34,409評論 4 333
  • 正文 年R本政府宣布季惯,位于F島的核電站,受9級特大地震影響勉抓,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜纵散,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,052評論 3 316
  • 文/蒙蒙 一隐圾、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蜜笤,春花似錦盐碱、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽聘惦。三九已至儒恋,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間禀酱,已是汗流浹背牧嫉。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留曹洽,地道東北人辽剧。 一個(gè)月前我還...
    沈念sama閱讀 46,503評論 2 361
  • 正文 我出身青樓,卻偏偏與公主長得像偷崩,于是被迫代替她去往敵國和親辟拷。 傳聞我的和親對象是個(gè)殘疾皇子阐斜,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,627評論 2 350

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