協(xié)程Python 中實(shí)現(xiàn)多任務(wù)耗資源最小的方式
協(xié)程,又稱微線程,纖程。英文名 Coroutine。
協(xié)程是 Python 中另外一種實(shí)現(xiàn)多任務(wù)的方式,只不過比線程更小,占用更小執(zhí)行單元(理解為需要的資源)。
為啥說它是一個(gè)執(zhí)行單元,因?yàn)樗詭?CPU 上下文。這樣只要在合適的時(shí)機(jī), 我們可以把一個(gè)協(xié)程 切換到另一個(gè)協(xié)程。 只要這個(gè)過程中保存或恢復(fù) CPU上下文那么程序還是可以運(yùn)行的。
通俗的理解:在一個(gè)線程中的某個(gè)函數(shù),可以在任何地方保存當(dāng)前函數(shù)的一些臨時(shí)變量等信息,然后切換到另外一個(gè)函數(shù)中執(zhí)行,注意不是通過調(diào)用函數(shù)的方式做到的,并且切換的次數(shù)以及什么時(shí)候再切換到原來的函數(shù)都由開發(fā)者自己確定。
協(xié)程和線程差異
在實(shí)現(xiàn)多任務(wù)時(shí), 線程切換從系統(tǒng)層面遠(yuǎn)不止保存和恢復(fù) CPU上下文這么簡(jiǎn)單。
操作系統(tǒng)為了程序運(yùn)行的高效性每個(gè)線程都有自己緩存 Cache 等等數(shù)據(jù),操作系統(tǒng)還會(huì)幫你做這些數(shù)據(jù)的恢復(fù)操作,所以線程的切換非常耗性能。
但是協(xié)程的切換只是單純的操作 CPU 的上下文,所以一秒鐘切換個(gè)上百萬次系統(tǒng)都抗得住。
之前我們講過 yield 關(guān)鍵字,現(xiàn)在就用它來實(shí)現(xiàn)多任務(wù)。
例子:
import time def task_1(): while True: print("--1--") time.sleep(0.5) yield def task_2(): while True: print("--2--") time.sleep(0.5) yield def main(): t1 = task_1() t2 = task_2() while True: next(t1) next(t2) if __name__ == "__main__": main()
運(yùn)行過程:
先讓 t1 運(yùn)行一會(huì),當(dāng) t1 遇到 yield 的時(shí)候,再返回到 main() 循環(huán)的地方,然后執(zhí)行 t2 , 當(dāng)它遇到 yield 的時(shí)候,再次切換到 t1 中,這樣 t1 和 t2 就交替運(yùn)行,最終實(shí)現(xiàn)了多任務(wù),協(xié)程。
運(yùn)行結(jié)果:
greenlet
為了更好使用協(xié)程來完成多任務(wù),Python 中的 greenlet 模塊對(duì)其封裝,從而使得切換任務(wù)變的更加簡(jiǎn)單。
首先你要安裝一下 greenlet 模塊。
pip3 install greenlet
from greenlet import greenlet import time def test1(): while True: print("---A--") gr2.switch() time.sleep(0.5) def test2(): while True: print("---B--") gr1.switch() time.sleep(0.5) gr1 = greenlet(test1) gr2 = greenlet(test2) # 切換到gr1中運(yùn)行 gr1.switch()
運(yùn)行結(jié)果:
和我們之前用 yield 實(shí)現(xiàn)的效果基本一樣,greenlet 其實(shí)是對(duì) yield 進(jìn)行了簡(jiǎn)單的封裝。
greenlet 實(shí)現(xiàn)多任務(wù)要比 yield 更簡(jiǎn)單,但是我們以后還是不用它。
上面例子中的延時(shí)是0.5秒,如果延遲是100秒,那么程序就會(huì)卡住100秒,就算有其他需要執(zhí)行的任務(wù),系統(tǒng)也不會(huì)切換過去,這100秒的時(shí)間是無法利用的。
這個(gè)問題下面來解決。
gevent
greenlet 已經(jīng)實(shí)現(xiàn)了協(xié)程,但是還是得進(jìn)行人工切換,是不是覺得太麻煩了。
Python 還有一個(gè)比 greenlet 更強(qiáng)大的并且能夠自動(dòng)切換任務(wù)的模塊 gevent。
gevent 是對(duì) greenlet 的再次封裝。
其原理是當(dāng)一個(gè) greenlet 遇到 IO(指的是input output 輸入輸出,比如網(wǎng)絡(luò)、文件操作等)操作時(shí),比如訪問網(wǎng)絡(luò),就自動(dòng)切換到其他的 greenlet,等到 IO 操作完成,再在適當(dāng)?shù)臅r(shí)候切換回來繼續(xù)執(zhí)行。
由于 IO 操作非常耗時(shí),經(jīng)常使程序處于等待狀態(tài),有了gevent 為我們自動(dòng)切換協(xié)程,就保證總有 greenlet 在運(yùn)行,而不是等待 IO。
首先還是得先安裝 gevent。
pip3 install gevent
例子:
import gevent def f(n): for i in range(n): print(gevent.getcurrent(), i) g1 = gevent.spawn(f, 3) g2 = gevent.spawn(f, 3) g3 = gevent.spawn(f, 3) g1.join() g2.join() g3.join()
運(yùn)行結(jié)果:
<Greenlet at 0x35aae40: f(3)> 0
<Greenlet at 0x35aae40: f(3)> 1
<Greenlet at 0x35aae40: f(3)> 2
<Greenlet at 0x374a780: f(3)> 0
<Greenlet at 0x374a780: f(3)> 1
<Greenlet at 0x374a780: f(3)> 2
<Greenlet at 0x374a810: f(3)> 0
<Greenlet at 0x374a810: f(3)> 1
<Greenlet at 0x374a810: f(3)> 2
可以看到,3個(gè) greenlet 是依次運(yùn)行而不是交替運(yùn)行。
這還無法判斷 gevent 是否實(shí)現(xiàn)了多任務(wù)的效果,最好的判斷情況是在運(yùn)行結(jié)果中 0 1 2 不按順序出現(xiàn)。
在 gevent 的概念中,我們提到 gevent 在遇到延時(shí)的時(shí)候會(huì)自動(dòng)切換任務(wù)。
那么,我們先給上面的例子添加延時(shí),再看效果。
import gevent import time def f(n): for i in range(n): print(gevent.getcurrent(), i) time.sleep(0.5) g1 = gevent.spawn(f, 3) g2 = gevent.spawn(f, 3) g3 = gevent.spawn(f, 3) g1.join() g2.join() g3.join()
運(yùn)行結(jié)果:
<Greenlet at 0x36aae40: f(3)> 0
<Greenlet at 0x36aae40: f(3)> 1
<Greenlet at 0x36aae40: f(3)> 2
<Greenlet at 0x384a780: f(3)> 0
<Greenlet at 0x384a780: f(3)> 1
<Greenlet at 0x384a780: f(3)> 2
<Greenlet at 0x384a810: f(3)> 0
<Greenlet at 0x384a810: f(3)> 1
<Greenlet at 0x384a810: f(3)> 2
在添加了延時(shí)之后,運(yùn)行結(jié)果并沒有改變。
其實(shí),gevent 要的不是 time.sleep() 的延時(shí),而是 gevent.sleep() 的延時(shí)。
import gevent def f(n): for i in range(n): print(gevent.getcurrent(), i) gevent.sleep(0.5) g1 = gevent.spawn(f, 3) g2 = gevent.spawn(f, 3) g3 = gevent.spawn(f, 3) g1.join() g2.join() g3.join()
join 還有一種更簡(jiǎn)單的寫法。
import time import gevent def f(n): for i in range(n): print(gevent.getcurrent(), i) gevent.sleep(0.5) gevent.joinall([ gevent.spawn(f, 3), gevent.spawn(f, 3), gevent.spawn(f, 3) ])
一般都是后面的這種寫法。
運(yùn)行結(jié)果:
<Greenlet at 0x2e5ae40: f(3)> 0
<Greenlet at 0x2ffa780: f(3)> 0
<Greenlet at 0x2ffa810: f(3)> 0
<Greenlet at 0x2e5ae40: f(3)> 1
<Greenlet at 0x2ffa780: f(3)> 1
<Greenlet at 0x2ffa810: f(3)> 1
<Greenlet at 0x2e5ae40: f(3)> 2
<Greenlet at 0x2ffa780: f(3)> 2
<Greenlet at 0x2ffa810: f(3)> 2
這下終于實(shí)現(xiàn)多任務(wù)的效果了, gevent 在遇到延時(shí)的時(shí)候,就自動(dòng)切換到其他任務(wù)。
這里是將 time 中的 sleep 換成了 gevent 中的 sleep。
那如果有網(wǎng)絡(luò)程序,網(wǎng)絡(luò)程序中也有許多堵塞,比如 connect, recv,accept,需要不需要換成 gevent 中的對(duì)應(yīng)方法。
理論上來說,是要換的。如果想用 gevent,那么就要把所有的延時(shí)操作,堵塞這一類的函數(shù),統(tǒng)統(tǒng)換成 gevent 中的對(duì)應(yīng)方法。
那有個(gè)問題,萬一我的代碼已經(jīng)寫了10萬行了,這換起來怎么破......
有什么辦法不需要手動(dòng)修改么,有,打個(gè)補(bǔ)丁即可。
import time import gevent from gevent import monkey # 有耗時(shí)操作時(shí)需要 # 將程序中用到的耗時(shí)操作的代碼,換為gevent中自己實(shí)現(xiàn)的模塊 monkey.patch_all() def f(n): for i in range(n): print(gevent.getcurrent(), i) time.sleep(0.5) g1 = gevent.spawn(f, 3) g2 = gevent.spawn(f, 3) g3 = gevent.spawn(f, 3) g1.join() g2.join() g3.join()
monkey.patch_all()
會(huì)自動(dòng)去檢查代碼,將所有會(huì)產(chǎn)生延時(shí)堵塞的方法,都自動(dòng)換成 gevent 中的方法。
運(yùn)行結(jié)果:
<Greenlet at 0x3dd91e0: f(3)> 0
<Greenlet at 0x3dd9810: f(3)> 0
<Greenlet at 0x3dd99c0: f(3)> 0
<Greenlet at 0x3dd91e0: f(3)> 1
<Greenlet at 0x3dd9810: f(3)> 1
<Greenlet at 0x3dd99c0: f(3)> 1
<Greenlet at 0x3dd91e0: f(3)> 2
<Greenlet at 0x3dd9810: f(3)> 2
<Greenlet at 0x3dd99c0: f(3)> 2
總結(jié):
通過利用延時(shí)的時(shí)間去做其他任務(wù),把時(shí)間都利用起來,這就是協(xié)程最大的意義。
到此這篇關(guān)于協(xié)程Python 中實(shí)現(xiàn)多任務(wù)耗資源最小的方式的文章就介紹到這了,更多相關(guān)Python多任務(wù)耗資源最小方式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
基于Django框架的rest_framework的身份驗(yàn)證和權(quán)限解析
Django 是一個(gè)基于 Python 的 Web 框架,可讓您快速創(chuàng)建高效的 Web 應(yīng)用程序,這篇文章主要介紹了基于Django框架的rest_framework的身份驗(yàn)證和權(quán)限解析,需要的朋友可以參考下2023-05-05Python設(shè)計(jì)模式結(jié)構(gòu)型組合模式
這篇文章主要介紹了Python設(shè)計(jì)模式結(jié)構(gòu)型組合模式,組合模式即Composite?Pattern,將對(duì)象組合成成樹形結(jié)構(gòu)以表示“部分-整體”的層次結(jié)構(gòu),組合模式使得用戶對(duì)單個(gè)對(duì)象和組合對(duì)象的使用具有一致性,下文具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-02-02python中matplotlib條件背景顏色的實(shí)現(xiàn)
這篇文章主要給大家介紹了關(guān)于python中matplotlib條件背景顏色的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用python具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09實(shí)例探究Python以并發(fā)方式編寫高性能端口掃描器的方法
端口掃描器就是向一批端口上發(fā)送請(qǐng)求來檢測(cè)端口是否打開的程序,這里我們以實(shí)例探究Python以并發(fā)方式編寫高性能端口掃描器的方法2016-06-06使用Python操作Excel中圖片的基礎(chǔ)示例(插入、替換、提取、刪除)
Excel是主要用于處理表格和數(shù)據(jù)的工具,我們也能在其中插入、編輯或管理圖片,為工作表增添視覺效果,提升報(bào)告的吸引力,本文將詳細(xì)介紹如何使用Python操作Excel中的圖片,文中有詳細(xì)代碼示例供大家參考,需要的朋友可以參考下2024-07-07Python中的命名元組簡(jiǎn)單而強(qiáng)大的數(shù)據(jù)結(jié)構(gòu)示例詳解
namedtuple是Python中一個(gè)非常有用的數(shù)據(jù)結(jié)構(gòu),它提供了一種簡(jiǎn)單的方式創(chuàng)建具有固定字段的輕量級(jí)對(duì)象,通過使用namedtuple,可以提高代碼的可讀性和可維護(hù)性,避免了使用類定義對(duì)象的復(fù)雜性,這篇文章主要介紹了Python中的命名元組簡(jiǎn)單而強(qiáng)大的數(shù)據(jù)結(jié)構(gòu),需要的朋友可以參考下2024-05-05Python實(shí)現(xiàn)RabbitMQ6種消息模型的示例代碼
這篇文章主要介紹了Python實(shí)現(xiàn)RabbitMQ6種消息模型的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03Python中的復(fù)制操作及copy模塊中的淺拷貝與深拷貝方法
淺拷貝和深拷貝是Python基礎(chǔ)學(xué)習(xí)中必須辨析的知識(shí)點(diǎn),這里我們將為大家解析Python中的復(fù)制操作及copy模塊中的淺拷貝與深拷貝方法:2016-07-07Python 分布式緩存之Reids數(shù)據(jù)類型操作詳解
這篇文章主要介紹了Python 分布式緩存之Reids數(shù)據(jù)類型操作詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06