您好,登錄后才能下訂單哦!
協程:又稱為微線程,英文名稱Coroutine。
作用:它擁有自己的寄存器上下文和棧,能保留上一次調用時的狀態,可以隨時暫停程序,隨時切換回來。
優點:
?無需線程上下文切換的開銷
?無需原子操作鎖定及同步的開銷
?方便切換控制流,簡化編程模型
?高并發+高擴展性+低成本:一個CPU支持上萬的協程都不是問題。所以很適合用于高并發處理
缺點:
?無法利用多核資源:協程的本質是個單線程,它不能同時將 單個CPU 的多個核用上,協程需要和進程配合才能運行在多CPU上
?進行阻塞(Blocking)操作(如IO時)會阻塞掉整個程序
使用yield實現協程:
def g(name): print("starting product ....") while True: baozi= yield # 程序暫停等待next print("{} is eating {}...".format(name,baozi)) def p(): conn.__next__() # 實例conn啟動yield conn2.__next__() # 實例conn2啟動yield i = 0 while i<5: i += 1 print("product: {}".format(i)) conn.send(i) # 向yield發送數據,yield恢復,并自動執行next conn2.send(i) # 向yield發送數據,yield恢復,并自動執行next conn = g("laoda") # 創建實例conn conn2 = g("laoer") # 創建實例conn2 product = p() # 啟動函數p
使用greenlet實現協程:不是python自帶模塊,需要安裝
創建協程對象的方法其實有兩個參數”greenlet(run=None, parent=None)”。參數”run”就是其要調用的方法,比如上例中的函數test1()和test2();參數”parent”定義了該協程對象的父協程,也就是說,greenlet協程之間是可以有父子關系的。如果不設或設為空,則其父協程就是程序默認的”main”主協程。
from greenlet import greenlet def test1(): print(1) t2.switch() # 函數暫停,切換到t2 print(2) def test2(): print(3) t1.switch() # 函數暫停,切換到t1 print(4) t1=greenlet(test1) # test1生成greenlet對象 t2=greenlet(test2) # test2生成greenlet對象 t1.switch()
使用gevent實現協程:第三方庫,需要安裝
實現了異步I/O,操作
方法 | 參數 | 作用 | 示例 |
spawn(func,func_args) | func:加入gevent的函數名 func_args:函數參數 | 把函數,創建協程實例 | |
joinall[spawn_list] | spawn_list:spawn方法列表 | 把創建的協程實例添加到異步列表 等待列表中的所有實例執行完畢 | |
sleep(time) | time:時間(秒) | 交出CPU控制權,時間為秒 | |
getcurrent() | 獲取當前協程內存地址 |
from gevent import gevent import random def creat(num): wait_time = random.Randomint(0,num) gevent.sleep(wait_time) print("{} wait done!".format(gevent.getcrrent())) gevent_list = [] for i in range(20): gevent_list.append(gevent.spawn(creat, i)) gevent.joinall(gevent_list) #Socket并發:(未測試) import gevent import socket class server(object): def __int__(self,ip,port,*args): self.server_in = socket.socket() self.server_in.bind(ip,port) self.server_in.listen(100) # self.spawn_list = [] def run(self): client_spawn = [] while True: conn,addr = self.server_in.accept() client_spawn.append(gevent.spawn(handler,conn)) gevent.joinall(client_spawn) def handler(self,conn): while True: recv_data = conn.recv(1024) if recv_data = 'exit': conn.shutdown(socket.SHUT_WR) break print("recv:".format(recv_data)) conn.send(recv_data.upper())
http://python.jobbole.com/86481/
協程詳解
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。