1. 程式人生 > 遊戲攻略 >《鬼玩人遊戲》劇情模式第二章任務指南 第二章任務完成方法

《鬼玩人遊戲》劇情模式第二章任務指南 第二章任務完成方法

python中yield的⽤法詳解——最簡單,最清晰的解釋(排序節省記憶體消耗)
⾸先我要吐槽⼀下,看程式的過程中遇見了yield這個關鍵字,然後百度的時候,發現沒有⼀個能簡單的讓我懂的,講起來真TM的都是頭頭
是道,什麼引數,什麼傳遞的,還⼝⼝聲聲說⾃⼰的教程是最簡單的,最淺顯易懂的,我就想問沒有有考慮過讀者的感受。


接下來是正題:
⾸先,如果你還沒有對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()的型別,它
已經是個<class 'range'>了,⽽不是⼀個list了,畢竟這個是需要優化的。
--------------------------------------------------------
作者:蒙德志鴿和資料大全
連結:https://wenku.baidu.com/view/a992ff2500d8ce2f0066f5335a8102d276a26113.html
來源:百度文庫
著作權歸作者所有。商業轉載請聯絡作者獲得授權,非商業轉載請註明出處。