成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術文章
文章詳情頁

在Python中如何使用yield

瀏覽:116日期:2022-06-17 10:59:17
一、生成器

如果在一個方法內,包含了 yield 關鍵字,那么這個函數就是一個「生成器」。

生成器其實就是一個特殊的迭代器,它可以像迭代器那樣,迭代輸出方法內的每個元素。

我們來看一個包含 yield 關鍵字的方法:

# coding: utf8# 生成器def gen(n): for i in range(n):yield ig = gen(5) # 創建一個生成器print(g)# <generator object gen at 0x10bb46f50>print(type(g)) # <type ’generator’># 迭代生成器中的數據for i in g: print(i) # Output:# 0 1 2 3 4

注意,在這個例子中,當我們執行 g = gen(5) 時,gen 中的代碼其實并沒有執行,此時我們只是創建了一個「生成器對象」,它的類型是 generator。

然后,當我們執行 for i in g,每執行一次循環,就會執行到 yield 處,返回一次 yield 后面的值。

這個迭代過程是和迭代器最大的區別。

換句話說,如果我們想輸出 5 個元素,在創建生成器時,這個 5 個元素其實還并沒有產生,什么時候產生呢?只有在執行for循環遇到 yield 時,才會依次生成每個元素。

此外,生成器除了和迭代器一樣實現迭代數據之外,還包含了其他方法:

generator.__next__():執行 for 時調用此方法,每次執行到 yield 就會停止,然后返回 yield 后面的值,如果沒有數據可迭代,拋出 StopIterator 異常,for 循環結束 generator.send(value):外部傳入一個值到生成器內部,改變 yield 前面的值 generator.throw(type[, value[, traceback]]):外部向生成器拋出一個異常 generator.close():關閉生成器

通過使用生成器的這些方法,我們可以完成很多有意思的功能。

二、next

先來看生成器的 __next__ 方法,我們看下面這個例子。

# coding: utf8def gen(n): for i in range(n):print(’yield before’)yield iprint(’yield after’)g = gen(3) # 創建一個生成器print(g.__next__()) # 0print(’----’)print(g.__next__()) # 1print(’----’)print(g.__next__()) # 2print(’----’)print(g.__next__()) # StopIteration# Output:# yield before# 0# ----# yield after# yield before# 1# ----# yield after# yield before# 2# ----# yield after# Traceback (most recent call last):# File 'gen.py', line 16, in <module># print(g.__next__()) # StopIteration# StopIteration

在這個例子中,我們定義了 gen 方法,這個方法包含了 yield 關鍵字。然后我們執行 g = gen(3) 創建一個生成器,但是這次沒有執行 for 去迭代它,而是多次調用 g.__next__() 去輸出生成器中的元素。

我們看到,當執行 g.__next__()時,代碼就會執行到 yield 處,然后返回 yield 后面的值,如果繼續調用 g.__next__(),注意,你會發現,這次執行的開始位置,是上次 yield 結束的地方,并且它還保留了上一次執行的上下文,繼續向后迭代。

這就是使用 yield 的作用,在迭代生成器時,每一次執行都可以保留上一次的狀態,而不是像普通方法那樣,遇到 return 就返回結果,下一次執行只能再次重復上一次的流程。

生成器除了能保存狀態之外,我們還可以通過其他方式,改變其內部的狀態,這就是下面要講的 send 和 throw 方法。

三、send

上面的例子中,我們只展示了在 yield 后有值的情況,其實還可以使用 j = yield i 這種語法,我們看下面的代碼:

# coding: utf8def gen(): i = 1 while True:j = yield ii *= 2if j == -1: break

此時如果我們執行下面的代碼:

for i in gen(): print(i) time.sleep(1)

輸出結果會是 1 2 4 8 16 32 64 ... 一直循環下去, 直到我們殺死這個進程才能停止。

這段代碼一直循環的原因在于,它無法執行到 j == -1 這個分支里 break 出來,如果我們想讓代碼執行到這個地方,如何做呢?

這里就要用到生成器的 send 方法了,send 方法可以把外部的值傳入生成器內部,從而改變生成器的狀態。

g = gen() # 創建一個生成器print(g.__next__()) # 1print(g.__next__()) # 2print(g.__next__()) # 4# send 把 -1 傳入生成器內部 走到了 j = -1 這個分支print(g.send(-1)) # StopIteration 迭代停止

當我們執行 g.send(-1) 時,相當于把 -1 傳入到了生成器內部,然后賦值給了 yield 前面的 j,此時 j = -1,然后這個方法就會 break 出來,不會繼續迭代下去。

四、throw

外部除了可以向生成器內部傳入一個值外,還可以傳入一個異常,也就是調用 throw 方法:

# coding: utf8def gen(): try:yield 1 except ValueError:yield ’ValueError’ finally:print(’finally’)g = gen() # 創建一個生成器print(g.__next__()) # 1# 向生成器內部傳入異常 返回ValueErrorprint(g.throw(ValueError))# Output:# 1# ValueError# finally

這個例子創建好生成器后,使用 g.throw(ValueError) 的方式,向生成器內部傳入了一個異常,走到了生成器異常處理的分支邏輯。

五、close

生成器的 close 方法也比較簡單,就是手動關閉這個生成器,關閉后的生成器無法再進行操作。

>>> g = gen()>>> g.close() # 關閉生成器>>> g.__next__() # 無法迭代數據Traceback (most recent call last): File '<stdin>', line 1, in <module>StopIteration

close 方法我們在開發中使用得比較少,了解一下就好。

六、使用場景

了解了 yield 和生成器的使用方式,那么 yield 和生成器一般用在哪些業務場景中呢?

下面我介紹幾個例子,分別是大集合的生成、簡化代碼結構、協程與并發,你可以參考這些使用場景來使用 yield。

大集合的生成

如果你想生成一個非常大的集合,如果使用 list 創建一個集合,這會導致在內存中申請一個很大的存儲空間,例如想下面這樣:

# coding: utf8def big_list(): result = [] for i in range(10000000000):result.append(i) return result# 一次性在內存中生成大集合 內存占用非常大for i in big_list(): print(i)

這種場景,我們使用生成器就能很好地解決這個問題。

因為生成器只有在執行到 yield 時才會迭代數據,這時只會申請需要返回元素的內存空間,代碼可以這樣寫:

# coding: utf8def big_list(): for i in range(10000000000):yield i# 只有在迭代時 才依次生成元素 減少內存占用for i in big_list(): print(i)

簡化代碼結構

我們在開發時還經常遇到這樣一種場景,如果一個方法要返回一個 list,但這個 list 是多個邏輯塊組合后才能產生的,這就會導致我們的代碼結構變得很復雜:

# coding: utf8def gen_list(): # 多個邏輯塊 組成生成一個列表 result = [] for i in range(10):result.append(i) for j in range(5):result.append(j * j) for k in [100, 200, 300]:result.append(k) return result for item in gen_list(): print(item)

這種情況下,我們只能在每個邏輯塊內使用 append 向 list 中追加元素,代碼寫起來比較??隆?/p>

此時如果使用 yield 來生成這個 list,代碼就簡潔很多:

# coding: utf8def gen_list(): # 多個邏輯塊 使用yield 生成一個列表 for i in range(10):yield i for j in range(5):yield j * j for k in [100, 200, 300]:yield kfor item in gen_list(): print(i)

使用 yield 后,就不再需要定義 list 類型的變量,只需在每個邏輯塊直接 yield 返回元素即可,可以達到和前面例子一樣的功能。

我們看到,使用 yield 的代碼更加簡潔,結構也更清晰,另外的好處是只有在迭代元素時才申請內存空間,降低了內存資源的消耗。

七、協程與并發

還有一種場景是 yield 使用非常多的,那就是「協程與并發」。

如果我們想提高程序的執行效率,通常會使用多進程、多線程的方式編寫程序代碼,最常用的編程模型就是「生產者-消費者」模型,即一個進程 / 線程生產數據,其他進程 / 線程消費數據。

在開發多進程、多線程程序時,為了防止共享資源被篡改,我們通常還需要加鎖進行保護,這樣就增加了編程的復雜度。

在 Python 中,除了使用進程和線程之外,我們還可以使用「協程」來提高代碼的運行效率。

什么是協程?

簡單來說,由多個程序塊組合協作執行的程序,稱之為「協程」。

而在 Python 中使用「協程」,就需要用到 yield 關鍵字來配合。

可能這么說還是太好理解,我們用 yield 實現一個協程生產者、消費者的例子:

# coding: utf8def consumer(): i = None while True:# 拿到 producer 發來的數據j = yield i print(’consume %s’ % j)def producer(c): c.__next__() for i in range(5):print(’produce %s’ % i)# 發數據給 consumerc.send(i) c.close()c = consumer()producer(c)# Output:# produce 0# consume 0# produce 1# consume 1# produce 2# consume 2# produce 3# consume 3...

這個程序的執行流程如下:

1.c = consumer() 創建一個生成器對象

2.producer(c) 開始執行,c.__next()__會啟動生成器 consumer 直到代碼運行到 j = yield i 處,此時 consumer 第一次執行完畢,返回

3.producer 函數繼續向下執行,直到 c.send(i)處,這里利用生成器的 send 方法,向 consumer 發送數據

4.consumer 函數被喚醒,從 j = yield i 處繼續開始執行,并且接收到 producer 傳來的數據賦值給 j,然后打印輸出,直到再次執行到 yield 處,返回

5.producer 繼續循環執行上面的過程,依次發送數據給 cosnumer,直到循環結束

6.最終 c.close() 關閉 consumer 生成器,程序退出

在這個例子中我們發現,程序在 producer 和 consumer 這 2 個函數之間來回切換執行,相互協作,完成了生產任務、消費任務的業務場景,最重要的是,整個程序是在單進程單線程下完成的。

到此這篇關于在Python中如何使用yield的文章就介紹到這了,更多相關yield的用法內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Python 編程
相關文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
99在线精品观看| 亚洲韩国精品一区| 亚洲毛片av在线| 欧美日韩精品久久| 国产午夜精品一区二区| 不卡免费追剧大全电视剧网站| 欧美日韩国产一二三| 日本不卡一区二区| 国产亚洲一级| 亚洲一区二区三区影院| 国产一区二区黄色| 亚洲精品国产品国语在线app| 一区免费在线| 国产精品九色蝌蚪自拍| 欧美日韩理论| 国产精品美女久久久久久久久久久| 欧美福利一区二区三区| 中文字幕精品—区二区四季| 欧美激情第8页| 国产喂奶挤奶一区二区三区| 91在线看国产| 国产精品久久久久9999吃药| 亚洲国产合集| 亚洲啪啪综合av一区二区三区| 亚洲国产综合在线看不卡| 综合自拍亚洲综合图不卡区| 夜久久久久久| 五月天亚洲婷婷| 欧美无乱码久久久免费午夜一区| 激情文学综合丁香| 日韩一区二区三区视频在线观看| 国产成人无遮挡在线视频| 日韩欧美一二三区| 成人高清伦理免费影院在线观看| 精品国产乱码久久久久久蜜臀| www.日韩av| 中文字幕国产一区| 亚洲深夜激情| 亚洲国产精品精华液网站| 久久夜色精品| 看片的网站亚洲| 91精品在线观看入口| 成人av在线播放网站| 国产精品网站导航| 一本色道久久综合亚洲精品不卡 | 欧美午夜不卡| 亚洲美女少妇撒尿| 亚洲综合欧美| 九一九一国产精品| 欧美v日韩v国产v| 狠狠88综合久久久久综合网| 一区二区三区视频在线看| 久久久久一区二区三区| 日韩高清在线电影| 91精品在线麻豆| 欧美日韩三级电影在线| 亚洲在线视频网站| 欧美日韩五月天| 99久久精品免费观看| 日韩毛片精品高清免费| 久久久99国产精品免费| 国产一区二区三区国产| 久久这里只精品最新地址| 亚洲国产精品一区在线观看不卡 | 成人小视频在线| **网站欧美大片在线观看| 久久午夜影视| 岛国精品在线播放| 1024成人网色www| 在线日韩国产精品| 99久久伊人久久99| 一区二区欧美精品| 精品视频999| 91视频在线观看免费| 亚洲国产一区二区三区| 欧美一级艳片视频免费观看| 欧美特黄一区| 午夜伦理一区二区| 日韩亚洲欧美中文三级| 亚洲国产精品第一区二区| 免费视频一区二区| 久久久综合激的五月天| 国产精品入口66mio| 国内精品视频一区二区三区八戒| 久久久久久久久久久久久女国产乱| 国产精品夜夜夜一区二区三区尤| 国产一区二区三区免费看| 亚洲欧洲韩国日本视频| 欧美亚一区二区| 午夜国产精品视频免费体验区| 视频一区二区中文字幕| 久久一夜天堂av一区二区三区 | 欧美一区视频| 日韩精品成人一区二区三区| 26uuu精品一区二区在线观看| 一本色道久久综合亚洲精品婷婷 | 久久精品久久精品| 国产精品久久久久影视| 欧美日韩综合一区| 欧美视频在线观看| 国产麻豆精品久久一二三| 亚洲欧美日韩在线不卡| 91精品国产综合久久福利软件| 亚洲精品影视| 成人听书哪个软件好| 亚洲国产精品天堂| 精品国产凹凸成av人导航| 色吧成人激情小说| 欧美在线亚洲综合一区| 精品在线免费视频| 中文字幕亚洲一区二区va在线| 欧美日韩日日骚| 亚洲精品孕妇| 成人app软件下载大全免费| 一区二区视频免费在线观看| 亚洲精品一区二区精华| 色爱区综合激月婷婷| 亚洲精选一区| 成人免费看片app下载| 日本成人在线看| 一区在线播放视频| 日韩午夜精品电影| 亚洲专区欧美专区| 91在线视频免费91| 日韩av不卡一区二区| 国产欧美一区二区精品性| 欧美日韩精品免费| 欧美亚洲免费在线| 欧美视频四区| 成人性生交大片免费看中文网站| 天堂在线亚洲视频| 中文字幕亚洲综合久久菠萝蜜| 日韩欧美中文字幕公布| 91传媒视频在线播放| 一本久道久久综合婷婷鲸鱼| 色综合天天综合色综合av| 激情六月婷婷久久| 午夜视频久久久久久| 国产精品国产三级国产a| 日韩欧美电影在线| 在线一区二区视频| 免费毛片一区二区三区久久久| 一区在线视频观看| 欧美69wwwcom| 99热在这里有精品免费| 国产91对白在线观看九色| 精品中文av资源站在线观看| 午夜欧美电影在线观看| 亚洲一区二区在线视频| 亚洲视频每日更新| 国产精品久久三| 中文字幕第一区综合| 久久亚洲精品国产精品紫薇| 91麻豆精品国产91| 欧美裸体一区二区三区| 欧美亚洲日本一区| 91精彩视频在线观看| 久久久www| 亚洲欧美国产不卡| 免费不卡亚洲欧美| 乱人伦精品视频在线观看| 国产精品综合色区在线观看| 亚洲免费激情| 亚洲三级国产| 亚洲国产高清一区| 日韩天天综合| 一本色道久久综合| 国产视频一区在线观看一区免费| 亚洲国产一区二区在线| 宅男噜噜噜66国产日韩在线观看| 精品1区2区3区4区| 一区在线视频| 在线看片一区| 99热在线精品观看| 国产精品久久久久久模特| 亚洲少妇诱惑| aa日韩免费精品视频一| 亚洲精品麻豆| 国产亚洲激情| 久久精品男女| 欧美三级电影在线看| 欧美日韩一区二区三区不卡| 欧美日韩国产bt| 欧美精品xxxxbbbb| 欧美一三区三区四区免费在线看 | 久久一本综合频道| 色婷婷香蕉在线一区二区| 91福利视频在线| 欧美日韩国产中文| 日韩一卡二卡三卡| 国产亚洲综合色| 中文一区一区三区高中清不卡| 中文字幕制服丝袜一区二区三区| 亚洲日本免费电影| 亚洲成人av一区| 日本欧美大码aⅴ在线播放| 久久精品国产第一区二区三区| 国产剧情在线观看一区二区| 成人性色生活片| 韩日成人av|