日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

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

淺談Python協程

瀏覽:162日期:2022-07-21 08:36:00

協程

協程,又稱微線程,纖程。英文名Coroutine。一句話說明什么是線程:協程是一種用戶態的輕量級線程。

協程擁有自己的寄存器上下文和棧。協程調度切換時,將寄存器上下文和棧保存到其他地方,在切回來的時候,恢復先前保存的寄存器上下文和棧。因此:

協程能保留上一次調用時的狀態(即所有局部狀態的一個特定組合),每次過程重入時,就相當于進入上一次調用的狀態,換種說法:進入上一次離開時所處邏輯流的位置。

協程的好處:

無需線程上下文切換的開銷 無需原子操作鎖定及同步的開銷 '原子操作(atomic operation)是不需要synchronized',所謂原子操作是指不會被線程調度機制打斷的操作;這種操作一旦開始,就一直運行到結束,中間不會有任何 context switch (切換到另一個線程)。原子操作可以是一個步驟,也可以是多個操作步驟,但是其順序是不可以被打亂,或者切割掉只執行部分。視作整體是原子性的核心。 方便切換控制流,簡化編程模型 高并發+高擴展性+低成本:一個CPU支持上萬的協程都不是問題。所以很適合用于高并發處理。

缺點:

無法利用多核資源:協程的本質是個單線程,它不能同時將 單個CPU 的多個核用上,協程需要和進程配合才能運行在多CPU上.當然我們日常所編寫的絕大部分應用都沒有這個必要,除非是cpu密集型應用。 進行阻塞(Blocking)操作(如IO時)會阻塞掉整個程序

使用yield實現協程操作例子

import timeimport queuedef consumer(name): print('--->starting eating baozi...') while True: new_baozi = yield print('[%s] is eating baozi %s' % (name, new_baozi)) # time.sleep(1)def producer(): # 生產者 r = con.__next__() r = con2.__next__() n = 0 while n < 5: n += 1 con.send(n) con2.send(n) print('033[32;1m[producer]033[0m is making baozi %s' % n)if __name__ == ’__main__’: con = consumer('c1') con2 = consumer('c2') p = producer()

程序執行的結果為:

--->starting eating baozi...--->starting eating baozi...[c1] is eating baozi 1[c2] is eating baozi 1[producer] is making baozi 1[c1] is eating baozi 2[c2] is eating baozi 2[producer] is making baozi 2[c1] is eating baozi 3[c2] is eating baozi 3[producer] is making baozi 3[c1] is eating baozi 4[c2] is eating baozi 4[producer] is making baozi 4[c1] is eating baozi 5[c2] is eating baozi 5[producer] is making baozi 5

問題來了,現在之所以能夠實現多并發的效果,是因為每一個生產者沒有任何花時間的代碼,所以他根本沒有卡住,如果這個時候在生產者這里sleep(1),那么速度一下子就變慢了,來看下下面的函數

def home(): print('in func 1') time.sleep(5) print('home exec done')

def bbs(): print('in func 2') time.sleep(2)

def login(): print('in func 2')

假如說nginx每次來一個請求都經過函數來處理,但它是一個單線程的情況,假如說nginx請求home頁,因為nginx在后臺處理是單線程,單線程的情況下同事過來三次請求,那該怎么辦?肯定是一次次的串行的執行啊,但是我為了讓他實現感覺是并發的效果,我是不是該在各個協程之間實行切換啊,但什么時候切換呢?那么,我問你,如果從一個請求進來直接打印一個print,那么我會在這個地方立刻切換嗎?因為這里面沒有任何的阻塞,不會被卡主,所以不需要立刻切換。如果他需要干一件事,比如整個home花了5s鐘,單線程是串行的,即便是使用了協程,那它還是串行的,為了保證并發的效果,什么時候進行切換?應該time.sleep(5)這里切換到bbs請求,那么bbs如果也sleep呢?那它就切換到下一個login,那么就是這么的切換。怎么才能實現一個單線程下實現上面程序的并發效果呢?就一句話,遇到io操作就切換,協程之所以能處理大并發,其實就是把io操作給擠掉了,就是io操作就切換,也就是這個程序只有CPU在運算,所以速度很快!那么問題又來了切換完之后,那么什么時候在切換回去啊?也就是說,怎么實現程序自動監測io操作完成了?那么就看下一個知識點吧!

Greenlet

greenlet是一個用C實現的協程模塊,相比與python自帶的yield,它是一塊封裝好了的協程,可以使你在任意函數之間隨意切換,而不需把這個函數先聲明為generator。

from greenlet import greenletdef test1(): print(12) gr2.switch() # 切換到gr2 print(34) gr2.switch() # 切換到gr2def test2(): print(56) gr1.switch() # 切換到gr1 print(78)gr1 = greenlet(test1) # 啟動一個協程gr2 = greenlet(test2) #gr1.switch() # 切換到gr1

程序執行后的結果為:

12563478

Gevent

上面的greenlet為手動擋的自動切換,現在來看一下自動擋的自動切換Gevent,遇到IO就切換。

Gevent 是一個第三方庫,可以輕松通過gevent實現并發同步或異步編程,在gevent中用到的主要模式是Greenlet, 它是以C擴展模塊形式接入Python的輕量級協程。 Greenlet全部運行在主程序操作系統進程的內部,但它們被協作式地調度。

來看下非常簡單的協程切換小程序

import geventdef func1(): print(’033[31;1m李闖在跟海濤搞...033[0m’) gevent.sleep(2) # 模仿IO print(’033[31;1m李闖又回去跟繼續跟海濤搞...033[0m’)def func2(): print(’033[32;1m李闖切換到了跟海龍搞...033[0m’) gevent.sleep(1) print(’033[32;1m李闖搞完了海濤,回來繼續跟海龍搞...033[0m’)gevent.joinall([ gevent.spawn(func1), # spawn 啟動一個協程 gevent.spawn(func2),])

程序執行后的結果為:

李闖在跟海濤搞...李闖切換到了跟海龍搞...李闖搞完了海濤,回來繼續跟海龍搞...李闖又回去跟繼續跟海濤搞...

協程之爬蟲

現在利用協程來實現簡單的爬蟲

from gevent import monkey; monkey.patch_all() # 把當前程序的所有的io操作單獨給我做上標記import gevent # 協程模塊from urllib.request import urlopen # 爬蟲所需要的模塊def f(url): print(’GET: %s’ % url) resp = urlopen(url) data = resp.read() print(’%d bytes received from %s.’ % (len(data), url))gevent.joinall([ # 利用協程大并發的爬取網頁 gevent.spawn(f, ’https://www.python.org/’), gevent.spawn(f, ’https://www.yahoo.com/’), gevent.spawn(f, ’https://github.com/’),])

程序執行的結果為:

GET: https://www.python.org/GET: https://www.yahoo.com/GET: https://github.com/59619 bytes received from https://github.com/.495691 bytes received from https://www.yahoo.com/.48834 bytes received from https://www.python.org/.

協程之Socket

通過gevent實現單線程下的多socket并發

# socket_server #import sysimport socketimport timeimport geventfrom gevent import socket,monkeymonkey.patch_all()def server(port): s = socket.socket() s.bind((’HW-20180425SPSL’, port)) s.listen(500) while True: cli, addr = s.accept() gevent.spawn(handle_request, cli)def handle_request(conn): try: while True: data = conn.recv(1024) print('recv:', data) conn.send(data) if not data: conn.shutdown(socket.SHUT_WR) except Exception as ex: print(ex) finally: conn.close()if __name__ == ’__main__’: server(8001)

# socket_client #import socketHOST = ’HW-20180425SPSL’ # The remote hostPORT = 8001 # The same port as used by the servers = socket.socket(socket.AF_INET, socket.SOCK_STREAM)s.connect((HOST, PORT))while True: msg = bytes(input('>>:'),encoding='utf8') s.sendall(msg) data = s.recv(1024) #print(data) print(’Received’, repr(data)) s.close()

程序執行后的結果為:

socket_client.py

>>:lalaReceived b’lala’>>:

socket_server.py

recv: b’heihei’

論事件驅動和異步IO

通常,我們寫服務器處理模型的程序時,有以下幾種模型:(1)每收到一個請求,創建一個新的進程,來處理該請求;

(2)每收到一個請求,創建一個新的線程,來處理該請求;

(3)每收到一個請求,放入一個事件列表,讓主進程通過非阻塞I/O方式來處理請求

上面的幾種方式,各有千秋,

第(1)中方法,由于創建新的進程的開銷比較大,所以,會導致服務器性能比較差,但實現比較簡單。

第(2)種方式,由于要涉及到線程的同步,有可能會面臨死鎖等問題。

第(3)種方式,在寫應用程序代碼時,邏輯比前面兩種都復雜。

綜合考慮各方面因素,一般普遍認為第(3)種方式是大多數網絡服務器采用的方式

看圖說話講事件驅動模型

在UI編程中,常常要對鼠標點擊進行相應,首先如何獲得鼠標點擊呢?

方式一:創建一個線程,該線程一直循環檢測是否有鼠標點擊,那么這個方式有以下幾個缺點:

1. CPU資源浪費,可能鼠標點擊的頻率非常小,但是掃描線程還是會一直循環檢測,這會造成很多的CPU資源浪費;如果掃描鼠標點擊的接口是阻塞的呢?

2. 如果是堵塞的,又會出現下面這樣的問題,如果我們不但要掃描鼠標點擊,還要掃描鍵盤是否按下,由于掃描鼠標時被堵塞了,那么可能永遠不會去掃描鍵盤;

3. 如果一個循環需要掃描的設備非常多,這又會引來響應時間的問題;所以,該方式是非常不好的。

方式二:就是事件驅動模型

目前大部分的UI編程都是事件驅動模型,如很多UI平臺都會提供onClick()事件,這個事件就代表鼠標按下事件。事件驅動模型大體思路如下:

1. 有一個事件(消息)隊列;

2. 鼠標按下時,往這個隊列中增加一個點擊事件(消息);

3. 有個循環,不斷從隊列取出事件,根據不同的事件,調用不同的函數,如onClick()、onKeyDown()等;

4. 事件(消息)一般都各自保存各自的處理函數指針,這樣,每個消息都有獨立的處理函數;

淺談Python協程

什么是事件驅動模型?

其實就是根據事件做出反應!

事件驅動編程是一種編程范式,這里程序的執行流由外部事件來決定。它的特點是包含一個事件循環,當外部事件發生時使用回調機制來觸發相應的處理。另外兩種常見的編程范式是(單線程)同步以及多線程編程。

讓我們用例子來比較和對比一下單線程、多線程以及事件驅動編程模型。下圖展示了隨著時間的推移,這三種模式下程序所做的工作。這個程序有3個任務需要完成,每個任務都在等待I/O操作時阻塞自身。阻塞在I/O操作上所花費的時間已經用灰色框標示出來了。

淺談Python協程

在單線程同步模型中,任務按照順序執行。如果某個任務因為I/O而阻塞,其他所有的任務都必須等待,直到它完成之后它們才能依次執行。這種明確的執行順序和串行化處理的行為是很容易推斷得出的。如果任務之間并沒有互相依賴的關系,但仍然需要互相等待的話這就使得程序不必要的降低了運行速度。

在多線程版本中,這3個任務分別在獨立的線程中執行。這些線程由操作系統來管理,在多處理器系統上可以并行處理,或者在單處理器系統上交錯執行。這使得當某個線程阻塞在某個資源的同時其他線程得以繼續執行。與完成類似功能的同步程序相比,這種方式更有效率,但程序員必須寫代碼來保護共享資源,防止其被多個線程同時訪問。多線程程序更加難以推斷,因為這類程序不得不通過線程同步機制如鎖、可重入函數、線程局部存儲或者其他機制來處理線程安全問題,如果實現不當就會導致出現微妙且令人痛不欲生的bug。

在事件驅動版本的程序中,3個任務交錯執行,但仍然在一個單獨的線程控制中。當處理I/O或者其他昂貴的操作時,注冊一個回調到事件循環中,然后當I/O操作完成時繼續執行。回調描述了該如何處理某個事件。事件循環輪詢所有的事件,當事件到來時將它們分配給等待處理事件的回調函數。這種方式讓程序盡可能的得以執行而不需要用到額外的線程。事件驅動型程序比多線程程序更容易推斷出行為,因為程序員不需要關心線程安全問題。

當我們面對如下的環境時,事件驅動模型通常是一個好的選擇:

1、程序中有許多任務,而且…

2、任務之間高度獨立(因此它們不需要互相通信,或者等待彼此)而且…

3、在等待事件到來時,某些任務會阻塞。

當應用程序需要在任務間共享可變的數據時,這也是一個不錯的選擇,因為這里不需要采用同步處理。

網絡應用程序通常都有上述這些特點,這使得它們能夠很好的契合事件驅動編程模型。

此處要提出一個問題,就是,上面的事件驅動模型中,只要一遇到IO就注冊一個事件,然后主程序就可以繼續干其它的事情了,只到io處理完畢后,繼續恢復之前中斷的任務,這本質上是怎么實現的呢?哈哈,下面我們就來一起揭開這神秘的面紗。。。。

請看詳解Python IO口多路復用這篇文章

以上就是淺談Python協程的詳細內容,更多關于Python協程的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
九九久久国产| 国产精品欧美在线观看| 国产一区二区三区四区二区| 欧美专区一区| 国产精品天天看天天狠| 久久超级碰碰| 国产精品毛片久久| 成人羞羞视频播放网站| 成人午夜精品| 97精品国产福利一区二区三区| 精品一区二区三区在线观看视频| 久久精品亚洲| 亚洲网站视频| 亚洲制服一区| 久久男人av| 久久91导航| 日韩欧美在线精品| 麻豆一区二区在线| 日韩另类视频| 亚洲+小说+欧美+激情+另类| 欧美日韩91| 麻豆精品视频在线观看视频| 久久国产生活片100| 久久国内精品| 亚洲欧美日韩高清在线| 美女视频一区在线观看| 久久99伊人| 久久精品1区| 国产极品嫩模在线观看91精品| 亚洲精品888| 91欧美国产| 精品视频亚洲| 国产美女高潮在线观看| 国产精品成人国产| 亚洲久草在线| 在线国产一区| 日韩精品欧美| 国产伦久视频在线观看| 国产精品中文| 亚洲一区国产| 美女黄网久久| 美女尤物久久精品| 亚洲一区二区毛片| 五月婷婷六月综合| 国内激情久久| 夜夜嗨一区二区| 久久国产精品亚洲77777| 香蕉精品视频在线观看| 亚洲电影在线| 婷婷亚洲五月| 亚洲成人精选| 欧美网站在线| 中文字幕一区二区精品区| 久久福利精品| 欧美理论视频| 日韩在线一区二区| 性欧美精品高清| 国产婷婷精品| 午夜欧美视频| 国产精选在线| 久久精品系列| 日韩久久精品网| 欧美不卡在线| 在线 亚洲欧美在线综合一区| 日本成人在线视频网站| 成人va天堂| 久久五月天小说| 樱桃成人精品视频在线播放| 91精品一区二区三区综合| 中文字幕日韩亚洲| 久久国内精品视频| 国产精品成人a在线观看| 国产精品欧美一区二区三区不卡| 精品国产美女a久久9999| 国产一区一一区高清不卡| 伊人精品一区| 午夜亚洲福利| 麻豆91小视频| 91精品国产乱码久久久久久久| 视频在线观看91| 色88888久久久久久影院| 老牛影视精品| 你懂的网址国产 欧美| 九色精品91| 欧美国产极品| 亚洲二区在线| 日韩精品91亚洲二区在线观看| 国产一区调教| 亚洲一区二区小说| 色在线视频观看| 久久中文亚洲字幕| 亚洲精选91| 久久亚洲人体| 三级在线看中文字幕完整版| 亚洲高清毛片| 日韩高清二区| 久久99久久人婷婷精品综合| 日韩有吗在线观看| 美女视频黄免费的久久| 亚洲欧美日韩专区| 亚洲欧洲专区| 7777精品| 成人国产精品一区二区免费麻豆| 97精品中文字幕| 亚洲福利专区| 久久精品欧洲| 日产欧产美韩系列久久99| 国产aⅴ精品一区二区四区| 亚洲国产日韩欧美在线| 在线视频亚洲| 91p九色成人| 午夜在线观看免费一区| 日韩高清不卡一区| 国产色99精品9i| 国产亚洲一区二区三区啪| 美女被久久久| 亚洲高清不卡| 日韩高清在线不卡| 夜夜精品视频| 欧美日韩亚洲一区| 午夜久久免费观看| 动漫av一区| 国产欧美丝祙| 中文字幕一区二区三区四区久久 | av一区二区高清| 午夜精品亚洲| 国产一区2区| 69堂精品视频在线播放| 婷婷激情一区| 国产成人免费| 国产伊人久久| 久久国产婷婷国产香蕉| 影音先锋久久精品| 久久久一二三| 国产成年精品| 91嫩草精品| 人人精品人人爱| 免费中文字幕日韩欧美| 久久这里只有精品一区二区| 亚洲欧美日韩在线观看a三区 | 鲁大师精品99久久久| 久久亚洲二区| 国产精品精品| 欧美日韩视频网站| 日韩a一区二区| 日本欧美国产| 久久亚洲精品中文字幕| 欧美亚洲色图校园春色| 成人羞羞在线观看网站| 亚洲欧美网站在线观看| 激情久久久久久| 热三久草你在线| 亚洲一区有码| 免播放器亚洲| 日韩中文欧美在线| 亚洲日本在线观看视频| 中文字幕日韩亚洲| 国产精品99一区二区| 国产精品99一区二区| 国产精品99免费看| 欧美国产另类| 欧美在线观看视频一区| 久久久精品久久久久久96| 美女国产精品| 丰满少妇一区| 欧美精品一区二区久久| 亚洲综合不卡| 精品视频在线你懂得| 奇米亚洲欧美| 国产福利一区二区三区在线播放| 国产日韩欧美在线播放不卡| 国产福利资源一区| 久久不卡日韩美女| 日韩综合一区| 合欧美一区二区三区| 国产精品一卡| 欧美福利专区| 久久久亚洲欧洲日产| 国产拍在线视频| 中文一区一区三区免费在线观| 欧美www视频在线观看| 日韩影院精彩在线| 国产一区三区在线播放| 欧美在线精品一区| 在线视频日韩| 激情欧美国产欧美| 青草国产精品| 欧美日韩黑人| 亚洲精品国产嫩草在线观看| 免费视频一区二区三区在线观看| 国产精品久一| 成人亚洲一区二区| 亚洲精品一级二级| 在线观看亚洲精品福利片| 国产视频一区二区在线播放| 精品深夜福利视频| 欧美精品九九| 欧美一级久久| 97精品视频在线看| 久久国产66|