欧美阿v视频在线大全_亚洲欧美中文日韩V在线观看_www性欧美日韩欧美91_亚洲欧美日韩久久精品

主頁 > 知識庫 > 在Python中如何使用yield

在Python中如何使用yield

熱門標簽:騰訊地圖標注沒法顯示 海外網吧地圖標注注冊 孝感營銷電話機器人效果怎么樣 打電話機器人營銷 南陽打電話機器人 ai電銷機器人的優勢 地圖標注自己和別人標注區別 聊城語音外呼系統 商家地圖標注海報

一、生成器

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

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

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

# coding: utf8

# 生成器
def gen(n):
    for i in range(n):
        yield i

g = 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: utf8

def gen(n):
    for i in range(n):
        print('yield before')
        yield i
        print('yield after')

g = gen(3)      # 創建一個生成器
print(g.__next__())  # 0
print('----')
print(g.__next__())  # 1
print('----')
print(g.__next__())  # 2
print('----')
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 就返回結果,下一次執行只能再次重復上一次的流程。

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

三、send

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

# coding: utf8

def gen():
    i = 1
    while True:
        j = yield i
        i *= 2
        if 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__())  # 1
print(g.__next__())  # 2
print(g.__next__())  # 4
# send 把 -1 傳入生成器內部 走到了 j = -1 這個分支
print(g.send(-1))   # StopIteration 迭代停止

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

四、throw

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

# coding: utf8

def gen():
    try:
        yield 1
    except ValueError:
        yield 'ValueError'
    finally:
        print('finally')

g = gen()   # 創建一個生成器
print(g.__next__()) # 1
# 向生成器內部傳入異常 返回ValueError
print(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: utf8

def big_list():
    result = []
    for i in range(10000000000):
        result.append(i)
    return result

# 一次性在內存中生成大集合 內存占用非常大
for i in big_list():
    print(i)

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

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

# coding: utf8

def big_list():
    for i in range(10000000000):
        yield i

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

簡化代碼結構

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

# coding: utf8

def 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)

這種情況下,我們只能在每個邏輯塊內使用 appendlist 中追加元素,代碼寫起來比較啰嗦。

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

# coding: utf8

def 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 k
        
for item in gen_list():
    print(i)

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

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

七、協程與并發

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

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

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

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

什么是協程?

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

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

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

# coding: utf8

def 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)
        # 發數據給 consumer
        c.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 生成器,程序退出

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

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

您可能感興趣的文章:
  • python如何正確使用yield
  • 通過實例簡單了解python yield使用方法
  • python使用yield壓平嵌套字典的超簡單方法
  • Python yield 使用方法淺析
  • Python yield的使用詳解

標簽:聊城 六盤水 楊凌 撫州 揚州 迪慶 牡丹江 南寧

巨人網絡通訊聲明:本文標題《在Python中如何使用yield》,本文關鍵詞  在,Python,中,如何,使用,yield,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《在Python中如何使用yield》相關的同類信息!
  • 本頁收集關于在Python中如何使用yield的相關信息資訊供網民參考!
  • 推薦文章
    欧美阿v视频在线大全_亚洲欧美中文日韩V在线观看_www性欧美日韩欧美91_亚洲欧美日韩久久精品
  • <rt id="w000q"><acronym id="w000q"></acronym></rt>
  • <abbr id="w000q"></abbr>
    <rt id="w000q"></rt>
    美脚の诱脚舐め脚责91| 中文字幕网站在线观看| 在线视频综合导航| 国产精品久久久久久久蜜臀| 精品一区二区三区久久| 精品久久久久久中文字幕人妻最新| 欧美精品xxxxbbbb| 亚洲一区二区三区中文字幕| 亚洲欧美激情一区二区三区| 欧美影视一区在线| 夜夜嗨av一区二区三区四季av| 日批视频在线看| 日本高清免费不卡视频| 亚洲精品中文字幕在线观看| 91免费版在线看| 欧美日韩免费视频| 午夜精品影院在线观看| 日韩aaaaa| 精品久久久久久最新网址| 久久国产三级精品| 免费看91的网站| 国产女人aaa级久久久级 | 国产精品二区一区二区aⅴ污介绍| 国产精品1区二区.| 黄色a级片在线观看| 日韩一区欧美小说| 色诱av手机版| 日韩一二在线观看| 久久99国产精品久久99果冻传媒| 少妇一级黄色片| 国产精品另类一区| 91老司机福利 在线| 555www色欧美视频| 久久99久国产精品黄毛片色诱| 亚洲欧洲综合网| **欧美大码日韩| 日本一区二区在线观看视频| 欧美r级在线观看| 国产福利一区在线| 欧美午夜电影一区| 日本aⅴ免费视频一区二区三区| 国产熟女一区二区| 中文字幕中文字幕一区| 女性生殖扒开酷刑vk| 26uuu亚洲婷婷狠狠天堂| 成人免费看视频| 欧美精选在线播放| 精品一区二区三区香蕉蜜桃| 久久久久亚洲AV成人| 亚洲成人动漫在线免费观看| 人人人妻人人澡人人爽欧美一区| 国产精品美女视频| 中国xxxx性xxxx产国| 精品国产乱码久久久久久老虎 | 亚洲欧美日韩久久| 捆绑裸体绳奴bdsm亚洲| 欧美国产激情一区二区三区蜜月| 91片黄在线观看| 精品乱人伦小说| 97久久精品人人爽人人爽蜜臀| 欧美一级xxx| eeuss鲁片一区二区三区| 欧美一区二区三区免费大片| 国产91精品欧美| 日韩一级大片在线| av爱爱亚洲一区| 精品国产免费久久| 中文字幕乱码在线人视频| 久久免费视频色| 日本少妇xxxx软件| 国产欧美精品一区| 一级特级黄色片| 亚洲色图19p| 亚洲精品成人av久久| 亚洲国产综合色| 顶级黑人搡bbw搡bbbb搡| 午夜亚洲国产au精品一区二区| 91麻豆精品成人一区二区| 日韩高清不卡在线| 日本精品一级二级| 国产老肥熟一区二区三区| 在线成人午夜影院| 成人a级免费电影| 久久综合久久综合久久| 亚洲av无码成人精品区| 国产精品日产欧美久久久久| 日本一区二区三区网站| 亚洲另类春色国产| 黑人と日本人の交わりビデオ| 亚洲国产精品一区二区久久恐怖片| 免费黄色激情视频| 日韩电影在线免费看| 欧美体内she精视频| 国产福利91精品一区二区三区| 欧美一区二视频| 在线观看你懂的视频| 国产精品久久久久一区| 欧美老女人性生活视频| 肉丝袜脚交视频一区二区| 欧美四级电影网| av在线这里只有精品| 日本一区二区免费在线| 人妻aⅴ无码一区二区三区| 亚州成人在线电影| 91福利区一区二区三区| 成人免费精品视频| 国产女同互慰高潮91漫画| 在线视频第一页| 青青草原综合久久大伊人精品 | 精品三级av在线| 91视频在线免费| 亚洲国产欧美另类丝袜| 欧美性受极品xxxx喷水| 成人av网址在线| 中文字幕在线不卡视频| 国产激情无码一区二区三区| 国产一本一道久久香蕉| 久久久青草青青国产亚洲免观| 深爱五月激情网| 日韩成人精品在线观看| 欧美一区二区久久| 喷水视频在线观看| 午夜国产精品一区| 在线电影国产精品| 漂亮人妻被黑人久久精品| 午夜电影久久久| 欧美一区中文字幕| 少妇饥渴放荡91麻豆| 日本不卡一区二区三区高清视频| 日韩三级视频在线看| 欧美性xxxx图片| 久久精品国产久精国产爱| 精品处破学生在线二十三| 国产毛片久久久久久久| 国内精品不卡在线| 国产欧美精品一区二区色综合 | 1024在线看片| 国产精品一区二区久久精品爱涩| 国产欧美日韩在线| 26uuu成人网| 99精品欧美一区| 亚洲亚洲人成综合网络| 91精品国产麻豆国产自产在线| 日韩av一二区| 国内国产精品久久| 国产精品久久久久影院亚瑟 | 日本在线观看网址| 粉嫩aⅴ一区二区三区四区五区| 国产精品福利av| 欧美日韩一区二区三区视频| 精品视频站长推荐| 激情综合一区二区三区| 国产精品美女久久久久aⅴ| 欧美中文字幕一区二区三区| 亚洲国产wwwccc36天堂| 制服.丝袜.亚洲.另类.中文| 亚洲久久久久久久| 国产精品12区| 亚洲精品乱码久久久久久黑人| 欧美精品成人一区二区三区四区| 成人免费网站黄| 粉嫩欧美一区二区三区高清影视| 亚洲欧美日韩久久精品| 欧美一区二区三区视频免费 | 一区二区三区四区免费| 国产大陆亚洲精品国产| 亚洲精品菠萝久久久久久久| 日韩一级黄色片| 国产午夜精品理论片在线| 欧美一区二区三区影院| 美女一区二区在线观看| 国产精品毛片高清在线完整版| 欧美私人免费视频| 日韩毛片无码永久免费看| av成人老司机| 蜜臀久久99精品久久久久久9| 国产精品萝li| 制服丝袜亚洲网站| 久久国产高清视频| 中文字幕天堂网| 国产成人三级在线观看| 亚洲国产精品一区二区www在线 | 91视频啊啊啊| 丰满少妇久久久久久久| 同产精品九九九| 国产精品毛片无遮挡高清| 911精品国产一区二区在线| 五月天免费网站| 亚洲自拍偷拍精品| 粉嫩绯色av一区二区在线观看| 亚洲成人动漫一区| 中文字幕中文字幕在线一区| 日韩欧美区一区二| 91久久香蕉国产日韩欧美9色| 国产人妻一区二区| av电影中文字幕| 国产suv精品一区二区883| 日韩精品亚洲专区| 亚洲免费观看高清完整版在线观看熊| 欧美va亚洲va国产综合|