mylist
是乙個可迭代的物件。當你使用乙個列表生成式來建立乙個列表的時候,就建立了乙個可迭代的物件:
mylist = [x*x for x in range(3)]
for i in mylist :
print(i)01
4
在這裡,所有的值都存在記憶體當中,所以並不適合大量資料
def creategenerator() :
mylist = range(3)
for i in mylist :
yield i*i
>>> mygenerator = creategenerator() # create a generator
>>> print(mygenerator) # mygenerator is an object!
for i in mygenerator:
print(i)01
4
第一次迭代中你的函式會執行,從開始到達 yield 關鍵字,然後返回 yield 後的值作為第一次迭代的返回值. 然後,每次執行這個函式都會繼續執行你在函式內部定義的那個迴圈的下一次,再返回那個值,直到沒有可以返回的。
先要理解迭代器,然後是生成器,最後要看yeild的功能,遇到yield會返回,但是返回的是乙個生成器。mygenerator = creategenerator() # create a generator 。
另外網上又看到一篇解釋的也不錯的文章:
如果你還沒有對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))
輸出
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.
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...12
3456
78910
Python的yield關鍵字
忽然得知python有個叫yield的關鍵字,好奇之下去查了查,花了點時間基本弄明白了,故寫在此備忘,順便充充字數。yield關鍵字用來定義生成器 generator 其具體功能是可以當return使用,從函式裡返回乙個值,不同之處是用yield返回之後,可以讓函式從上回yield返回的地點繼續執行...
python 理解關鍵字 yield
為了方便追 更新擼了乙個基於scrapy的爬蟲。在實現過程中使用到了yield,網上對其的文字描述都很難讓人理解。通過debug 才了解呼叫順序,進而理解了它使用方法。我們可以用乙個等式來形容其作用 yeild 函式 return 生成器 generator 用同步方式寫非同步 生成器 可迭代的函式...
Python的yield關鍵字
忽然得知python有個叫yield的關鍵字,好奇之下去查了查,花了點時間基本弄明白了,故寫在此備忘,順便充充字數。yield關鍵字用來定義生成器 generator 其具體功能是可以當return使用,從函式裡返回乙個值,不同之處是用yield返回之後,可以讓函式從上回yield返回的地點繼續執行...