摘要:程序執行,程序會從關鍵字那一行繼續向下運行,會把這個值賦值給變量由于方法中包含方法,所以程序會繼續向下運行執行方法,然后再次進入循環程序執行再次遇到關鍵字,會返回后面的值后,程序再次暫停,直到再次調用方法或方法。
此文轉載,侵刪,原文地址:https://blog.csdn.net/mieleiz...
首先,如果你還沒有對yield有個初步分認識,那么你先把yield看做“return”,這個是直觀的,它首先是個return,普通的return是什么意思,就是在程序中返回某個值,返回之后程序就不再往下運行了。看做return之后再把它看做一個是生成器(generator)的一部分(帶yield的函數才是真正的迭代器),好了,如果你對這些不明白的話,那先把yield看做return,然后直接看下面的程序,你就會明白yield的全部意思了:
def foo(): print("starting...") while True: res = yield 4 print("res:",res) g = foo() print(next(g)) print("*"*20) print(next(g))
就這么簡單的幾行代碼就讓你明白什么是yield,代碼的輸出這個:
starting... 4 ******************** res: None 4
我直接解釋代碼運行順序,相當于代碼單步調試:
1.程序開始執行以后,因為foo函數中有yield關鍵字,所以foo函數并不會真的執行,而是先得到一個生成器g(相當于一個對象)
2.直到調用next方法,foo函數正式開始執行,先執行foo函數中的print方法,然后進入while循環
3.程序遇到yield關鍵字,然后把yield想想成return,return了一個4之后,程序停止,并沒有執行賦值給res操作,此時next(g)語句執行完成,所以輸出的前兩行(第一個是while上面的print的結果,第二個是return出的結果)是執行print(next(g))的結果,
4.程序執行print(""20),輸出20個*
5.又開始執行下面的print(next(g)),這個時候和上面那個差不多,不過不同的是,這個時候是從剛才那個next程序停止的地方開始執行的,也就是要執行res的賦值操作,這時候要注意,這個時候賦值操作的右邊是沒有值的(因為剛才那個是return出去了,并沒有給賦值操作的左邊傳參數),所以這個時候res賦值是None,所以接著下面的輸出就是res:None,
6.程序會繼續在while里執行,又一次碰到yield,這個時候同樣return 出4,然后程序停止,print函數輸出的4就是這次return出的4.
?
到這里你可能就明白yield和return的關系和區別了,帶yield的函數是一個生成器,而不是一個函數了,這個生成器有一個函數就是next函數,next就相當于“下一步”生成哪個數,這一次的next開始的地方是接著上一次的next停止的地方執行的,所以調用next的時候,生成器并不會從foo函數的開始執行,只是接著上一步停止的地方開始,然后遇到yield后,return出要生成的數,此步就結束。
def foo(): print("starting...") while True: res = yield 4 print("res:",res) g = foo() print(next(g)) print("*"*20) print(g.send(7))
再看一個這個生成器的send函數的例子,這個例子就把上面那個例子的最后一行換掉了,輸出結果:
starting... 4 ******************** res: 7 4
先大致說一下send函數的概念:此時你應該注意到上面那個的紫色的字,還有上面那個res的值為什么是None,這個變成了7,到底為什么,這是因為,send是發送一個參數給res的,因為上面講到,return的時候,并沒有把4賦值給res,下次執行的時候只好繼續執行賦值操作,只好賦值為None了,而如果用send的話,開始執行的時候,先接著上一次(return 4之后)執行,先把7賦值給了res,然后執行next的作用,遇見下一回的yield,return出結果后結束。
?
5.程序執行g.send(7),程序會從yield關鍵字那一行繼續向下運行,send會把7這個值賦值給res變量
6.由于send方法中包含next()方法,所以程序會繼續向下運行執行print方法,然后再次進入while循環
7.程序執行再次遇到yield關鍵字,yield會返回后面的值后,程序再次暫停,直到再次調用next方法或send方法。
這就結束了,說一下,為什么用這個生成器,是因為如果用List的話,會占用更大的空間,比如說取0,1,2,3,4,5,6............1000
你可能會這樣:
for n in range(1000): a=n
這個時候range(1000)就默認生成一個含有1000個數的list了,所以很占內存。
這個時候你可以用剛才的yield組合成生成器進行實現,也可以用xrange(1000)這個生成器實現
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時已經沒有xrange()了,在python3中,range()就是xrange()了,你可以在python3中查看range()的類型,它已經是個
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/43228.html
摘要:使用單引號雙引號和三引號或來表示字符串。不可變的集合函數會以字典類型返回當前位置的全部全局變量。用于將進制整數轉換成進制,以字符串形式表示。返回字符串中最大的字母,或數組中的最大值。的作用就是減少了單行函數的定義。 問題答案由本人整理 1.基礎語法是否熟悉?介紹一下 Python和其他語言最大的區別就是使用行和縮進,而不是大括號({})或者分號(;)來控制類、函數或者邏輯判斷。Pyt...
摘要:于此同時,會阻塞,等待終止。子生成器返回之后,解釋器會拋出異常,并把返回值附加到異常對象上,只是委派生成器恢復。實例運行完畢后,返回的值綁定到上。這一部分處理調用方通過方法傳入的異常。之外的異常會向上冒泡。 上一篇python協程1:yield的使用介紹了: 生成器作為協程使用時的行為和狀態 使用裝飾器預激協程 調用方如何使用生成器對象的 .throw(...) 和 .close()...
摘要:是的一個關鍵字,剛接觸的時候對這個關鍵字一知半解,掌握之后才發現這關鍵字有大用,本文將對的使用方法好好梳理一番。使用創建生成器在中,生成器是一種可迭代對象,但可迭代對象不一定是生成器。 yield是python的一個關鍵字,剛接觸python的時候對這個關鍵字一知半解,掌握之后才發現這關鍵字有大用,本文將對yield的使用方法好好梳理一番。 1 使用yield創建生成器 在python...
摘要:通過創建將所有的異步操作邏輯收集在一個地方集中處理,可以用來代替中間件。 redux-saga框架使用詳解及Demo教程 前面我們講解過redux框架和dva框架的基本使用,因為dva框架中effects模塊設計到了redux-saga中的知識點,可能有的同學們會用dva框架,但是對redux-saga又不是很熟悉,今天我們就來簡單的講解下saga框架的主要API和如何配合redux框...
閱讀 3882·2021-09-23 11:51
閱讀 3067·2021-09-22 15:59
閱讀 868·2021-09-09 11:37
閱讀 2070·2021-09-08 09:45
閱讀 1267·2019-08-30 15:54
閱讀 2065·2019-08-30 15:53
閱讀 492·2019-08-29 12:12
閱讀 3290·2019-08-29 11:15