91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》
  • 首頁 > 
  • 教程 > 
  • 開發技術 > 
  • Python協程操作之gevent(yield阻塞,greenlet),協程實現多任務(有規律的交替協作執行)用法詳解

Python協程操作之gevent(yield阻塞,greenlet),協程實現多任務(有規律的交替協作執行)用法詳解

發布時間:2020-09-27 10:37:10 來源:腳本之家 閱讀:207 作者:houyanhua1 欄目:開發技術

本文實例講述了Python 協程操作之gevent(yield阻塞,greenlet),協程實現多任務(有規律的交替協作執行)用法。分享給大家供大家參考,具體如下:

實現多任務:進程消耗的資源最大,線程消耗的資源次之,協程消耗的資源最少(單線程)。

gevent實現協程,gevent是通過阻塞代碼(例如網絡延遲等)來自動切換要執行的任務,所以在進行IO密集型程序時(例如爬蟲),使用gevent可以提高效率(有效利用網絡延遲的時間去執行其他任務)。

GIL(全局解釋器鎖)是C語言版本的Python解釋器中專有的,GIL的存在讓多線程的效率變低(哪個線程搶到鎖,就執行哪個線程)。在IO密集型程序中,多線程依然比單線程效率高(GIL通過IO阻塞自動切換多線程)。

解決GIL(全局解釋器鎖)的問題的三種方法:1、不要用C語言版本的Python解釋器。2、讓子線程運行其他語言代碼(例如:主線程運行Python代碼,子線程運行C語言代碼(C語言的動態庫))。3、多進程代替多線程(多進程可以利用多核CPU)。

demo.py(協程底層原理,yield):

import time
# 帶yield的函數并不是函數,而是一個生成器模板,返回一個生成器(可用于遍歷迭代)
def task_1():
  while True:
    time.sleep(0.1)
    yield # 阻塞,等待next迭代解阻塞
def task_2():
  while True:
    time.sleep(0.1)
    yield
def main():
  t1 = task_1() # 返回的t1是一個生成器。 此時并未執行task_1中的代碼
  t2 = task_2()
  # 先讓t1運行一會,當t1中遇到yield的時候,再返回到24行,然后
  # 執行t2,當它遇到yield的時候,再次切換到t1中
  # 這樣t1/t2/t1/t2的交替運行,最終實現了多任務....協程
  while True:
    next(t1) # 執行task_1中的代碼,遇到yield阻塞task_1。 等待下一次next激活。
    next(t2) # 實現task_1與task_2交替執行,實現多任務...協程
if __name__ == "__main__":
  main()

demo.py(greenlet實現協程,封裝了yield):

from greenlet import greenlet # 需要安裝greenlet模塊 sudo pip3 install greenlet (python2.x使用pip) 
import time
def test1():
  while True:
    print("---A--")
    gr2.switch()  # 切換到gr2中的任務。
    time.sleep(0.5)
def test2():
  while True:
    print("---B--")
    gr1.switch()  # 切換到gr1中的任務。
    time.sleep(0.5)
gr1 = greenlet(test1) # greenlet 底層封裝了yield。
gr2 = greenlet(test2)
#切換到gr1中運行
gr1.switch()

demo.py(gevent實現協程,封裝了greenlet,遇到阻塞代碼自動切換協程任務):

import gevent # 需要安裝gevent模塊 sudo pip3 install gevent (python2.x使用pip) 
import time
def f1(n):
  for i in range(n):
    print(gevent.getcurrent(), i)
    gevent.sleep(0.5)  # 為了提高協程效率,遇到阻塞類代碼,會自動切換協程任務。
    # time.sleep(0.5)  # 阻塞類代碼必須使用gevent自己包裝的代碼,原生阻塞類代碼不會切換協程任務。 
              # 可以使用monkey.patch_all()將所有原生阻塞類代碼替換成gevent包裝的阻塞類代碼。 
def f2(n):
  for i in range(n):
    print(gevent.getcurrent(), i) # <Greenlet "Greenlet-0" at 0x7f4a09b34648: f1(5)> 0
    gevent.sleep(0.5)
    # time.sleep(0.5)
def f3(n):
  for i in range(n):
    print(gevent.getcurrent(), i)
    gevent.sleep(0.5)
    # time.sleep(0.5)
g1 = gevent.spawn(f1, 5) # gevent其實是對greenlet的封裝。
g2 = gevent.spawn(f2, 5) # 第一個參數f2表示協程執行的具體任務(函數),第二個參數5表示要傳給f2的參數
g3 = gevent.spawn(f3, 5)
g1.join()  # 遇到阻塞類代碼,自動切換協程任務。
g2.join()
g3.join()

運行結果:

<Greenlet at 0x1985030: f1(5)> 0
<Greenlet at 0x1b431c8: f2(5)> 0
<Greenlet at 0x1b43140: f3(5)> 0
<Greenlet at 0x1985030: f1(5)> 1
<Greenlet at 0x1b431c8: f2(5)> 1
<Greenlet at 0x1b43140: f3(5)> 1
<Greenlet at 0x1985030: f1(5)> 2
<Greenlet at 0x1b431c8: f2(5)> 2
<Greenlet at 0x1b43140: f3(5)> 2
<Greenlet at 0x1985030: f1(5)> 3
<Greenlet at 0x1b431c8: f2(5)> 3
<Greenlet at 0x1b43140: f3(5)> 3
<Greenlet at 0x1985030: f1(5)> 4
<Greenlet at 0x1b431c8: f2(5)> 4
<Greenlet at 0x1b43140: f3(5)> 4

demo.py(gevent打補丁,monkey自動替換原生阻塞類代碼。重要,常用):

import gevent # 需要安裝gevent模塊 sudo pip3 install gevent (python2.x使用pip)
import time
from gevent import monkey
# gevent打補丁
monkey.patch_all() # 將所有原生阻塞類代碼自動替換成gevent包裝的阻塞類代碼。 
def f1(n):
  for i in range(n):
    print(gevent.getcurrent(), i)
    time.sleep(0.5) # 會自動替換成 gevent.sleep(0.5)
def f2(n):
  for i in range(n):
    print(gevent.getcurrent(), i)
    time.sleep(0.5)
def f3(n):
  for i in range(n):
    print(gevent.getcurrent(), i)
    time.sleep(0.5)
# g1 = gevent.spawn(f1, 5)
# g2 = gevent.spawn(f2, 5)
# g3 = gevent.spawn(f3, 5)
# g1.join()
# g2.join()
# g3.join()
# 一種簡便寫法
gevent.joinall([
    gevent.spawn(f1, 5),
    gevent.spawn(f2, 5),
    gevent.spawn(f3, 5)
])

運行結果:

<Greenlet at 0x22e0938: f1(5)> 0
<Greenlet at 0x22e09c0: f2(5)> 0
<Greenlet at 0x22e0a48: f3(5)> 0
<Greenlet at 0x22e0938: f1(5)> 1
<Greenlet at 0x22e09c0: f2(5)> 1
<Greenlet at 0x22e0a48: f3(5)> 1
<Greenlet at 0x22e0938: f1(5)> 2
<Greenlet at 0x22e09c0: f2(5)> 2
<Greenlet at 0x22e0a48: f3(5)> 2
<Greenlet at 0x22e0938: f1(5)> 3
<Greenlet at 0x22e09c0: f2(5)> 3
<Greenlet at 0x22e0a48: f3(5)> 3
<Greenlet at 0x22e0938: f1(5)> 4
<Greenlet at 0x22e09c0: f2(5)> 4
<Greenlet at 0x22e0a48: f3(5)> 4

demo.py(gevent底層原理):

import socket
import time
tcp_server_tcp = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
tcp_server_tcp.bind(("", 7899))
tcp_server_tcp.listen(128)
tcp_server_tcp.setblocking(False) # 設置套接字為非堵塞的方式。 (接收數據時如果沒有接到數據(阻塞)那么就拋異常,否則正常接收數據。)
client_socket_list = list() # 用于保存與客戶端連接的套接字。
while True:
  # time.sleep(0.5)
  try:
    new_socket, new_addr = tcp_server_tcp.accept() # 用拋異常的方式代替阻塞。
  except Exception as ret:
    print("---沒有新的客戶端到來---")
  else:
    print("---只要沒有產生異常,那么也就意味著 來了一個新的客戶端----")
    new_socket.setblocking(False) # 設置套接字為非堵塞的方式。 (如果需要阻塞就直接拋異常代替阻塞)
    client_socket_list.append(new_socket)
  for client_socket in client_socket_list:
    try:
      recv_data = client_socket.recv(1024)  # 用拋異常的方式代替阻塞。
    except Exception as ret:
      print("----這個客戶端還沒有發送過來數據----")
    else:
      if recv_data:
        # 對方發送過來數據
        print("----客戶端發送過來了數據-----")
      else:
        # 對方調用close 導致了 recv返回
        client_socket.close()
        client_socket_list.remove(client_socket)
        print("---客戶端已經關閉----")

更多關于Python相關內容感興趣的讀者可查看本站專題:《Python進程與線程操作技巧總結》、《Python數據結構與算法教程》、《Python函數使用技巧總結》、《Python字符串操作技巧匯總》、《Python入門與進階經典教程》、《Python+MySQL數據庫程序設計入門教程》及《Python常見數據庫操作技巧匯總》

希望本文所述對大家Python程序設計有所幫助。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

兴业县| 大石桥市| 泰兴市| 济阳县| 冷水江市| 铁岭市| 伊春市| 康平县| 定结县| 兴山县| 司法| 白朗县| 阿鲁科尔沁旗| 金川县| 瑞安市| 博湖县| 瑞昌市| 四川省| 阿瓦提县| 泰顺县| 开江县| 道真| 崇文区| 乾安县| 偏关县| 嘉兴市| 辽阳县| 稷山县| 昌吉市| 凤庆县| 恩施市| 阜南县| 高碑店市| 元江| 桂林市| 庐江县| 方山县| 霍城县| 邵东县| 温泉县| 三门县|