国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

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

Python實現異步IO的示例

瀏覽:25日期:2022-07-06 11:12:26

前言

用阻塞 API 寫同步代碼最簡單,但一個線程同一時間只能處理一個請求,有限的線程數導致無法實現萬級別的并發連接,過多的線程切換也搶走了 CPU 的時間,從而降低了每秒能夠處理的請求數量。為了達到高并發,你可能會選擇一個異步框架,用非阻塞 API 把業務邏輯打亂到多個回調函數,通過多路復用與事件循環的方式實現高并發。

磁盤 IO 為例,描述了多線程中使用阻塞方法讀磁盤,2 個線程間的切換方式。那么,怎么才能實現高并發呢?

Python實現異步IO的示例

把上圖中本來由內核實現的請求切換工作,交由用戶態的代碼來完成就可以了,異步化編程通過應用層代碼實現了請求切換,降低了切換成本和內存占用空間。異步化依賴于 IO 多路復用機制,比如 Linux 的 epoll 或者 Windows 上的 iocp,同時,必須把阻塞方法更改為非阻塞方法,才能避免內核切換帶來的巨大消耗。Nginx、Redis 等高性能服務都依賴異步化實現了百萬量級的并發。

下圖描述了異步 IO 的非阻塞讀和異步框架結合后,是如何切換請求的。

Python實現異步IO的示例

然而,寫異步化代碼很容易出錯。因為所有阻塞函數,都需要通過非阻塞的系統調用拆分成兩個函數。雖然這兩個函數共同完成一個功能,但調用方式卻不同。第一個函數由你顯式調用,第二個函數則由多路復用機制調用。

這種方式違反了軟件工程的內聚性原則,函數間同步數據也更復雜。特別是條件分支眾多、涉及大量系統調用時,異步化的改造工作會非常困難。

Python如何實現異步調用

from flask import Flaskimport timeapp = Flask(__name__)@app.route(’/bar’)def bar(): time.sleep(1) return ’<h1>bar!</h1>’@app.route(’/foo’)def foo(): time.sleep(1) return ’<h1>foo!</h1>’if __name__ == ’__main__’: app.run(host=’127.0.0.1’,port=5555,debug=True)

采用同步的方式調用

import requestsimport timestarttime = time.time()print(requests.get(’http://127.0.0.1:5555/bar’).content)print(requests.get(’http://127.0.0.1:5555/foo’).content)print('消耗時間: ',time.time() -starttime)

b’<h1>bar!</h1>’b’<h1>foo!</h1>’消耗時間: 2.015509605407715

采樣異步的方式調用:

重點:

1.將阻塞io改為非阻塞io;

2.多路復用io監聽內核事件,事件觸發通過回調函數;

3.用戶態代碼采取事件循環的方式獲取事件,執行事件的回調函數;

import selectorsimport socketimport time# from asynrequest import ParserHttpclass asynhttp: def __init__(self): self.selecter = selectors.DefaultSelector() def get(self,url,optiondict = None): global reqcount reqcount += 1 s = socket.socket() s.setblocking(False) try: s.connect((’127.0.0.1’,5555)) except BlockingIOError: pass requset = ’GET %s HTTP/1.0rnrn’ % url callback = lambda : self.send(s,requset) self.selecter.register(s.fileno(),selectors.EVENT_WRITE,callback) def send(self,s,requset): self.selecter.unregister(s.fileno()) s.send(requset.encode()) chunks = [] callback = lambda: self.recv(s,chunks) self.selecter.register(s.fileno(),selectors.EVENT_READ,callback) def recv(self,s,chunks): self.selecter.unregister(s.fileno()) chunk = s.recv(1024) if chunk: chunks.append(chunk) callback = lambda: self.recv(s,chunks) self.selecter.register(s.fileno(), selectors.EVENT_READ, callback) else: global reqcount reqcount -= 1 request_first,request_headers,request_content,_ = ParserHttp.parser(b’’.join(chunks)) print('解析數據:',request_first,request_headers,request_content) print((b’’.join(chunks)).decode()) return (b’’.join(chunks)).decode()starttime = time.time()reqcount = 0asynhttper = asynhttp()asynhttper.get(’/bar’)asynhttper.get(’/foo’)while reqcount: events = asynhttper.selecter.select() for event,mask in events: func = event.data func()print('消耗時間:' ,time.time() - starttime)

HTTP/1.0 200 OKContent-Type: text/html; charset=utf-8Content-Length: 13Server: Werkzeug/1.0.1 Python/3.7.7Date: Thu, 15 Oct 2020 03:28:16 GMT

<h1>bar!</h1>HTTP/1.0 200 OKContent-Type: text/html; charset=utf-8Content-Length: 13Server: Werkzeug/1.0.1 Python/3.7.7Date: Thu, 15 Oct 2020 03:28:16 GMT

<h1>foo!</h1>消耗時間: 1.0127637386322021

以上就是Python實現異步IO的示例的詳細內容,更多關于python 異步IO的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 99ri在线精品视频在线播放 | 99国内精品 | 欧美另类 videos黑人极品 | 另类视频综合 | 亚洲成a人片 | 亚洲美女在线观看播放 | 日韩毛片高清免费 | 国产精品久久久久久亚洲伦理 | 成人欧美视频在线观看 | 欧美国产永久免费看片 | 精品91一区二区三区 | 波多野一区二区三区在线 | 国产高清在线观看视频手机版 | 亚洲综合精品 | 日本特黄a级高清免费酷网 日本特黄特色 | 波多野结衣中文无毒不卡 | 国产午夜永久福利视频在线观看 | 最刺激黄a大片免费网站 | 亚洲国产精品a一区二区三区 | 香港a毛片免费全部播放 | 国产一区二区三区不卡免费观看 | 免费视频一区二区 | 成人精品一区二区激情 | 久久免费看视频 | 偷拍精品视频一区二区三区 | 国产成人刺激视频在线观看 | 色色视频免费网 | 精品国产v无码大片在线观看 | 韩国一区在线 | 亚洲精品影院久久久久久 | 一级毛片免费观看视频 | 国产一级在线 | 91精品免费国产高清在线 | 亚洲aa| 欧美日产国产亚洲综合图区一 | 亚洲成人看片 | 久久99精品国产免费观看 | 亚洲精品色一区二区三区 | 欧美丰满大乳大屁股毛片 | 欧美色性| 香蕉亚洲精品一区二区 |