淺談Python非同步編程

来源:https://www.cnblogs.com/beyond-tester/archive/2023/10/25/17786087.html
-Advertisement-
Play Games

1. 非同步編程概述 非同步編程是一種編程範式,用於處理那些需要等待I/O操作完成或者耗時任務的情況。在傳統的同步編程中,代碼會按照順序逐行執行,直到遇到一個耗時操作,它會阻塞程式的執行直到操作完成。這種阻塞式的模型在某些場景下效率低下,因為代碼在等待操作完成時無法執行其他任務。 非同步編程通過使用非阻塞 ...


1. 非同步編程概述

非同步編程是一種編程範式,用於處理那些需要等待I/O操作完成或者耗時任務的情況。在傳統的同步編程中,代碼會按照順序逐行執行,直到遇到一個耗時操作,它會阻塞程式的執行直到操作完成。這種阻塞式的模型在某些場景下效率低下,因為代碼在等待操作完成時無法執行其他任務。

非同步編程通過使用非阻塞I/O和協程(coroutine)來提高效率。協程是一種特殊的函數,可以在執行過程中暫停和恢復。當一個協程遇到一個耗時操作時,它會暫停自己的執行,讓出控制權給其他協程,從而實現併發執行。async/await關鍵字是Python中處理協程的語法工具

2. async/await關鍵字

async關鍵字

async是一個關鍵字,用於定義一個協程函數。協程函數可以通過使用await關鍵字來暫停自身的執行,等待其他協程或非同步操作完成。

以下是一個簡單的示例,展示瞭如何定義一個協程函數:

import asyncio

async def my_coroutine():
    print("Coroutine started")
    await asyncio.sleep(1)
    print("Coroutine resumed")
    return "Result"

my_coroutine是一個協程函數。它使用了async關鍵字進行定義,並包含了一個await語句來暫停執行。

await關鍵字

await是另一個關鍵字,用於暫停協程函數的執行,等待另一個協程、非同步操作或者Future對象完成。

以下是一個使用await的示例:

import asyncio

async def my_coroutine():
    print("Coroutine started")
    await asyncio.sleep(1)
    print("Coroutine resumed")
    return "Result"

async def main():
    result = await my_coroutine()
    print(f"Result: {result}")

asyncio.run(main())

在上面的示例中,main函數是一個協程函數,它使用await關鍵字來等待my_coroutine協程函數的執行結果。當await語句執行時,main函數會暫停自身的執行,直到my_coroutine協程函數完成並返回結果。

需要註意的是,await關鍵字只能在協程函數中使用。如果你在普通的同步函數中使用await,會導致語法錯誤。

3. 非同步事件迴圈

非同步編程的核心是事件迴圈(event loop)。事件迴圈負責調度和執行協程,確保它們按照正確的順序執行。

在Python中,可以使用asyncio模塊提供的事件迴圈來創建和管理協程。

以下是一個使用事件迴圈的示例:

import asyncio

async def my_coroutine():
    print("Coroutine started")
    await asyncio.sleep(1)
    print("Coroutine resumed")
    return "Result"

async def main():
    result = await my_coroutine()
    print(f"Result: {result}")

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

在上面的示例中,asyncio.get_event_loop()用於獲取預設的事件迴圈對象。然後,通過調用run_until_complete方法來運行main協程函數,直到它完成

非同步編程最常見的用例是處理I/O操作,例如讀寫文件或與網路通信。在傳統的同步編程中,這些操作會阻塞程式的執行,直到操作完成。而在非同步編程中,可以使用非同步IO操作來實現非阻塞的併發執行。

4. 非同步IO操作

Python提供了asyncio模塊來處理非同步IO操作。asyncio中的一些常用函數和類包括:

  • asyncio.sleep(delay): 創建一個休眠指定時間的協程。
  • asyncio.open_connection(host, port): 創建一個協程,用於與指定的主機和埠建立網路連接。
  • asyncio.open_unix_connection(path): 創建一個協程,用於與指定路徑的UNIX域套接字建立連接。
  • asyncio.start_server(client_connected_cb, host, port): 創建一個協程,用於監聽指定主機和埠的連接請求,併在每次連接時調用client_connected_cb回調函數。

以下是一個使用非同步IO操作的示例:

import asyncio

async def read_data():
    # 模擬非同步IO讀取操作
    await asyncio.sleep(1)
    return "Data"

async def write_data(data):
    # 模擬非同步IO寫入操作
    await asyncio.sleep(1)
    print(f"Data written: {data}")

async def main():
    data = await read_data()
    await write_data(data)

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

在上面的示例中,read_datawrite_data函數模擬了非同步的IO讀取和寫入操作。在main函數中,我們使用await關鍵字等待讀取操作完成,然後將結果傳遞給寫入操作。

執行步驟如下:

  1. 首先,創建一個事件迴圈(Event Loop)對象,使用asyncio.get_event_loop()獲取預設的事件迴圈。

  2. 定義了三個協程函數:read_data()write_data()main()

  3. 調用loop.run_until_complete(main()),將main()協程任務提交給事件迴圈並運行,直到main()協程完成。

  4. main()協程中,首先調用read_data()協程函數。這會啟動read_data()協程,併在await asyncio.sleep(1)處暫停執行,等待1秒鐘。

  5. 在暫停執行的同時,事件迴圈可以切換到其他可運行的協程,例如write_data()協程。

  6. write_data()協程同樣會啟動,併在await asyncio.sleep(1)處暫停執行,等待1秒鐘。

  7. write_data()協程暫停執行時,事件迴圈沒有其他可運行的協程,因此它會等待,直到有其他協程可運行。

  8. 在等待1秒鐘後,read_data()協程恢復執行。它完成後,返回結果"Data"。

  9. main()協程接收到read_data()協程的返回結果,將其賦值給data變數。

  10. main()協程繼續執行,調用write_data(data)協程。

  11. write_data()協程恢復執行,列印出"data"的值。

  12. main()協程完成,事件迴圈結束。

在這個過程中,通過使用await關鍵字,協程能夠在等待IO操作完成時暫停執行,並允許事件迴圈切換到其他協程。這種方式下,IO操作可以以非同步的方式執行,而不會阻塞整個程式的執行流程。

5. 併發執行多個協程

非同步編程的一個關鍵優勢是能夠併發執行多個協程,以提高程式的性能。

asyncio提供了多種方式來實現協程的併發執行,其中最常用的方式是使用asyncio.gather函數。

以下是一個併發執行多個協程的示例:

import asyncio

async def coroutine1():
    await asyncio.sleep(1)
    print("Coroutine 1 completed")

async def coroutine2():
    await asyncio.sleep(2)
    print("Coroutine 2 completed")

async def coroutine3():
    await asyncio.sleep(0.5)
    print("Coroutine 3 completed")

async def main():
    await asyncio.gather(coroutine1(), coroutine2(), coroutine3())

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

在上面的示例中,coroutine1coroutine2coroutine3是三個協程函數。在main函數中,我們使用asyncio.gather函數來併發執行這三個協程。asyncio.gather接受一個可變數量的協程參數,並返回一個新的協程,該協程在所有給定的協程完成後完成。執行循序為當執行到 coroutine1中的await時,此協程會掛起,執行權交給新的協程 coroutine2開始執行,以此類推。當 coroutine3 等待0.5s執行完畢後,執行權重新回到coroutine3 ,繼續執行一下語句,其他同理。

需要註意的是,Python的協程是單線程的,通過事件迴圈來實現併發執行。當一個協程遇到阻塞的IO操作時,它會暫停自身的執行,並切換到下一個可執行的協程。這種切換是由事件迴圈調度


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在js中,js變數和JSON是兩個不同數據格式,兩者的儲存方式自然不相同。JSON格式是一種數據交換的規則,js變數則是javascript在程式需求場景中的數據表示。在js與不同語言的服務端進行數據交換過程中,js能夠有內置的方法將其變數轉化為JSON格式。 JSON.parse(data);// ...
  • 我們是袋鼠雲數棧 UED 團隊,致力於打造優秀的一站式數據中台產品。我們始終保持工匠精神,探索前端道路,為社區積累並傳播經驗價值。 本文作者:的盧 引入 在日常開發過程中,我們會使用很多性能優化的 API,比如像使用 memo、useMemo優化組件或者值,再比如使用 shouldComponent ...
  • 把Code Review 作為開發流程的必選項後,不代表Code Review這件事就可以執行的很好,因為Code Review 的執行,很大部分程度上依賴於審查者的認真審查,以及被審查者的積極配合,兩者缺一不可! ...
  • 寫在前面 react整體是函數式的思想,把組件設計成純組件,狀態和邏輯通過參數傳入,而vue的思想是響應式的,也就是基於是數據可變的,通過對每一個屬性建立Watcher來監聽, 當屬性變化的時候,響應式的更新對應的虛擬dom react的思路通過js來生成html, 所以設計了jsx,還有通過js來 ...
  • 接上一節:從零用VitePress搭建博客教程(6) -– 第三方組件庫的使用和VitePress搭建組件庫文檔 我們搭建完成vitePress後,那麼接下來就是如何部署到線上伺服器,這裡使用Github Pages,免得自己購買伺服器,當然你也可以自己購買伺服器來部署(比如阿裡雲伺服器)。 在部署 ...
  • Telegram 宣佈為其開發者提供了一項“能夠在 App 中運行迷你應用”的新功能( 迷你應用即 Mini App,下文中以“小程式”代替)。 ...
  • 一、定義 動態地給一個對象增加一些額外的職責。就擴展功能而言,裝飾模式提供了一種比使用子類更加靈活的替代方案。裝飾模式是一種結構型模式。 二、描述 包含以下三個角色:1、Component(抽象構件):它是具體構件和抽象裝飾類的父類,聲明瞭在具體構件中實現的業務方法,它的引入可以是客戶端以一致的方式 ...
  • 幾年前,我被問到“你是如何變成一名架構師的?”。基於這個話題,我們討論了很多,比如必要的技術、經驗以及所需要的知識儲備等。這一次討論促使我開始思考要成為一名架構師應該具備和學習的東西有哪些,成為一個優秀的架構師應該具備哪些能力和做哪些事情。為此我查閱資料,走訪各位大佬,當然也結合自己的經歷,最終我輸 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...