」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 使用 Asyncio 進行非同步編程

使用 Asyncio 進行非同步編程

發佈於2024-08-21
瀏覽:183

Asynchronous Programming with Asyncio

在编程世界中,“非阻塞”的概念无处不在。 JavaScript 开发人员经常使用术语“异步”,因为它是 JavaScript 的优势之一。然而,要真正理解异步编程,必须掌握并发和并行编程的概念。

并发编程

当多个独立实体同时工作时,编程是并发的。这并不一定意味着这些任务在完全相同的时间运行。相反,它意味着任务通过共享资源(例如 CPU 时间)随着时间的推移不断取得进展。并发编程的主要优点是它的鲁棒性:如果一个进程崩溃,程序的其余部分继续运行。

并行编程

如果一个算法可以将其工作分成几个部分,那么它就是并行的。拥有的处理器越多,您从并行性中受益就越多。高效的并行编程可优化现代机器的资源以获得更好的性能。

用 Cooking 说明并发与并行

并发示例:

想象一下,您正在准备一顿饭,需要烤一些肉并制作酱汁。首先将肉放在烤架上。当肉烤的时候,你可以把西红柿和其他蔬菜切碎作为酱汁。然后,你开始煮酱汁,同时偶尔检查一下肉。在这里,两项任务(烤肉和制作酱汁)都在进行中,但您正在它们之间切换注意力。这代表并发。

并行度示例:

现在,假设您有一位朋友可以帮助您。当您专注于烤肉时,您的朋友则负责制作酱汁。这两项任务同时完成,无需在它们之间切换注意力。这代表并行性。

什么是异步编程?

异步编程涉及处理程序外部发生的输入/输出 (I/O) 操作,例如用户输入、打印到终端、从套接字读取或写入磁盘。异步 I/O 的关键特性是:

  • 操作所花费的时间与CPU无关。相反,它取决于磁盘速度、网络延迟和其他外部条件等因素。

  • 程序无法预测操作何时结束。

对于具有大量 I/O 的服务(如 Web 服务器、数据库和部署脚本),优化这些操作可以极大地提高性能。

让我们看看阻塞代码和非阻塞代码的示例。

阻塞和非阻塞代码示例

考虑一个简单的程序:

import time

def task():
    time.sleep(2)
    print("Hello")

for _ in range(3):
    task()

在这个同步程序中,每个任务都会等待前一个任务完成,从而导致延迟。

现在,让我们看一下使用 asyncio 的异步版本:

import asyncio

async def task():
    await asyncio.sleep(2)
    print("Hello")

async def main():
    tasks = [task() for _ in range(3)]
    await asyncio.gather(*tasks)

asyncio.run(main())

在这个异步程序中,任务同时运行,减少了总执行时间。让我们探索一下异步编程的组成部分。

异步编程的组成部分

事件循环、协程和 future 是异步 Python 程序的基本元素。

  • 事件循环:管理任务切换和执行流程,跟踪要异步运行的任务。

  • 协程: 可以暂停和恢复的特殊功能,允许在等待期间运行其他任务。协程指定任务切换事件应在函数中发生的位置,将控制权返回给事件循环。协程通常由事件循环创建并内部存储在任务队列中。

  • Futures: 协程结果的占位符,存储结果或异常。一旦事件循环启动一个协程,就会创建一个相应的 future 来存储协程的结果,如果在协程执行期间抛出异常,则会创建一个异常。

解释完 Python 异步编程的关键部分后,让我们编写一些代码。

编写异步代码

现在您已经了解了异步编程模式,让我们编写一个小脚本并分析执行情况。这是一个简单的异步脚本:

import asyncio

async def task():
    await asyncio.sleep(2)
    print("Hello")

async def main():
    tasks = [task() for _ in range(3)]
    await asyncio.gather(*tasks)

asyncio.run(main())

在上面的代码中,即使另一个正在执行的任务正在休眠(阻塞),我们也尝试继续执行其他任务。注意任务和主要函数前面的 async 关键字。

这些函数现在是协程

Python 中的协程函数前面带有关键字 async。这里的 main() 函数是任务协调器或我们的单个事件循环,因为它使用 async.gather 方法执行所有任务。 asyncio.gather 函数同时运行可等待对象。

输出:

Hello
Hello
Hello
Program executed in 2.01 seconds.

当每个任务到达await asyncio.sleep(2)时,它只是转到下一个任务并在完成时返回。这就像说,“我要睡 2 秒钟。做点别的事。”

让我们看看同步版本以便快速比较。

import time

def task():
    time.sleep(2)
    print("Hello")

for _ in range(3):
    task()

在上面的代码中,我们将采用传统的Python编程方式。您会注意到该过程的执行将花费更多时间。

输出:

Hello
Hello
Hello
Program executed in 6.01 seconds.

现在您可以注意到执行时间了。将 time.sleep() 视为阻塞任务,将 asyncio.sleep() 视为非阻塞或长时间任务。在异步编程中,等待某些内容(例如 asyncio.sleep())的好处是,周围的函数可以暂时将控制权交给另一个准备立即执行的函数。

了解了 Python 异步编程的一些基本示例后,让我们探索一下 Python 异步编程的规则。

异步编程规则

  1. 协程:协程不能直接执行。如果您尝试直接运行协程函数,它将返回一个协程对象。相反,使用 asyncio.run():

    import asyncio
    
    async def hello():
        await asyncio.sleep(1)
        print('Hello')
    
    asyncio.run(hello())
    
  2. 可等待对象: 协程、future 和任务是主要的可等待对象。 Python 协程是可等待的,并且可以被其他协程等待。

  3. Await 关键字:await 只能在异步函数中使用。

    async def hello():
        await asyncio.sleep(1)
        print("Hello")
    
  4. 兼容性:并非所有Python模块都与异步编程兼容。例如,将await asyncio.sleep() 替换为time.sleep() 将导致错误。您可以在此处查看兼容和维护的模块列表。

在下一节中,我们将探讨异步编程的常见用法,HTTP 请求。

程序示例:异步请求

我们看下面这段代码:

import aiohttp
import asyncio

async def fetch(session, city):
    url = f"https://www.prevision-meteo.ch/services/json/{city}"
    async with session.get(url) as response:
        data = await response.json()
        print(f"Temperature at {city}: {data['current_condition']['tmp']} C")

async def main():
    async with aiohttp.ClientSession() as session:
        cities = ['paris', 'toulouse', 'marseille']
        tasks = [fetch(session, city) for city in cities]
        await asyncio.gather(*tasks)

asyncio.run(main())

在上面的代码中,我们创建了两个异步函数:一个用于从 prevision-meteo URL 获取数据,另一个主函数用于执行 Python 代码中的进程。目标是发送异步 HTTP GET 请求来检索温度并打印响应。

在main和fetch函数中,我们使用async with。在 fetch 函数中,async with 确保连接正确关闭。在 main 函数中,它确保 ClientSession 在完成请求后关闭。这些实践对于 Python 中的异步编码以有效管理资源并防止泄漏非常重要。

在main函数的最后一行,我们使用await asyncio.gather(*tasks)。在我们的例子中,它同时运行所有任务,允许程序同时发送多个 HTTP 请求。使用await 可确保程序等待所有任务完成后再继续。

输出:

Temperature at marseille: 25 C
Temperature at toulouse: 24 C
Temperature at paris: 18 C
Program executed in 5.86 seconds.

用于比较的同步版本

代码:

import requests
import time

def fetch(city):
    url = f"https://www.prevision-meteo.ch/services/json/{city}"
    response = requests.get(url)
    data = response.json()
    print(f"Temperature at {city}: {data['current_condition']['tmp']} C")

def main():
    cities = ['paris', 'toulouse', 'marseille']
    for city in cities:
        fetch(city)

start_time = time.time()
main()
print(f"Program executed in {time.time() - start_time:.2f} seconds.")

输出:

Temperature at Paris: 18 C
Temperature at Toulouse: 24 C
Temperature at Marseille: 25 C
Program executed in 9.01 seconds.

何时使用异步编程

异步模型在以下情况下表现最佳:

  • 任务数量较多,保证至少有一个任务可以一直进行。

  • 任务涉及大量 I/O,导致异步程序在其他任务可以运行时浪费大量时间进行阻塞。

  • 任务在很大程度上是独立的,最大限度地减少任务间通信(因此一个任务等待另一任务)。

结论

在本教程中,我们介绍了:

  • 异步编程的概念及相关概念。

  • async/await的有效使用。

  • 使用 aiohttp 发出异步 HTTP 请求。

  • 异步编程的好处。

感谢您的阅读。第二部分将介绍 Django 的异步编程。

资源

  • Python 文档:协程和任务

  • Python 文档:asyncio - 异步 I/O

  • aiohttp 文档

  • Aio 库

  • 并发与并行

版本聲明 本文轉載於:https://dev.to/koladev/asynchronous-programming-with-asyncio-3ad1?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • ## 您可以在不使用 JavaScript 的情況下使用 CSS 建立餅圖分段嗎?
    ## 您可以在不使用 JavaScript 的情況下使用 CSS 建立餅圖分段嗎?
    使用 CSS 在圓中分段使用 border-radius 在 CSS 中建立圓是一種常見的做法。但是,我們可以透過分段(類似餅圖)來實現類似的效果嗎?本文深入研究了僅透過 HTML 和 CSS 實現此目的的方法,不包括使用 JavaScript。 產生相等大小的段相等大小段的一種方法涉及產生以下內容...
    程式設計 發佈於2024-11-06
  • 從頭開始建立一個小型向量存儲
    從頭開始建立一個小型向量存儲
    With the evolving landscape of generative AI, vector databases are playing crucial role in powering generative AI applications. There are so many vect...
    程式設計 發佈於2024-11-06
  • 如何在Chrome使用AI實驗API
    如何在Chrome使用AI實驗API
    要在 Chrome 中使用實驗性 AI API,請依照下列步驟操作: 硬體需求 4GB 記憶體 GPU可用 至少 22GB 空間 Windows 10.11 或 macOS Ventura 或更新版本(無 Linux 規格) 尚不支持: Chrome作業系統 Chrome iOS C...
    程式設計 發佈於2024-11-06
  • 評論:Adam Johnson 的《Boost Your Django DX》
    評論:Adam Johnson 的《Boost Your Django DX》
    書評很微妙。您不想破壞它,但您也想讓潛在讀者體驗所期待的內容。這是提供背景和保持興趣之間的巧妙平衡。我試圖在這篇評論中達到這種平衡,為您提供足夠的內容來吸引您,而不透露太多。 一個小背景故事:我第一次從 Djangonaut Space 的好朋友 Tim 那裡聽說這本書,並將其添加到我的閱讀清單中...
    程式設計 發佈於2024-11-06
  • 如何將陣列元素分組並組合多維數組中另一列的值?
    如何將陣列元素分組並組合多維數組中另一列的值?
    按列將數組元素分組並組合另一列中的值給定一個包含兩列嵌套數組的數組,任務是將基於特定列的子數組,並將每個組中另一列的值連接起來,產生以逗號分隔的清單。 考慮以下範例陣列:$array = [ ["444", "0081"], ["44...
    程式設計 發佈於2024-11-06
  • 三個新加入的例外功能
    三個新加入的例外功能
    從 JDK 7 開始,異常處理已擴展為三個新功能:自動資源管理、多重捕獲和更準確的重新拋出。 多重catch可讓您使用同一個catch子句擷取多個異常,避免程式碼重複。 要使用多重捕獲,請指定由 | 分隔的異常清單。在 catch 子句中。每個參數都是隱式最終參數。 用法範例:catch(f...
    程式設計 發佈於2024-11-06
  • 如何修復執行 ES6 程式碼時出現「意外的令牌匯出」錯誤?
    如何修復執行 ES6 程式碼時出現「意外的令牌匯出」錯誤?
    「排除意外的令牌匯出錯誤」嘗試在專案中執行ES6 程式碼時,可能會出現「意外的令牌導出”錯誤。此錯誤表示所使用的環境不支援 ES6 模組中使用的匯出關鍵字語法。 錯誤詳細資料以下程式碼片段舉例說明了錯誤的來源: export class MyClass { constructor() { ...
    程式設計 發佈於2024-11-06
  • 即使卸載後,VSCode 擴充功能也不會從檔案系統中刪除,我建立了一個解決方案!
    即使卸載後,VSCode 擴充功能也不會從檔案系統中刪除,我建立了一個解決方案!
    所以這是基於 vscode 的編輯器的問題。即使您卸載了擴充功能,它也會保留在檔案系統中,並隨著時間的推移堵塞您的系統。我創建了一個簡單的解決方案。執行此 python 腳本將刪除 vscode 上未安裝的擴充功能。 它適用於 VS Code、VS Code Insiders,也適用於 VSCod...
    程式設計 發佈於2024-11-06
  • 透過 GitHub Actions 按計畫更新網站內容
    透過 GitHub Actions 按計畫更新網站內容
    我想分享我建立一個自我永續的內容管理系統的旅程,該系統不需要傳統意義上的內容資料庫。 問題 該網站的內容(部落格文章和書籤)儲存在 Notion 資料庫中: 附書籤的資料庫 –  Notion UI 我試圖解決的問題是不必在添加每個書籤後手動部署網站。最重要的是 - 保持託管盡可能...
    程式設計 發佈於2024-11-06
  • 如何在 Laravel 5 應用程式的共享託管環境中清除快取?
    如何在 Laravel 5 應用程式的共享託管環境中清除快取?
    如何從 Laravel 5 中的共享託管伺服器清除快取清除快取對於維護 Laravel 應用程式的效能和效率至關重要。但是,在您可能無法存取 CLI 的共享託管環境中,清除快取可能是一個挑戰。 清除視圖快取的解決方法在這種情況下,您可以透過在 CLI 之外呼叫 Artisan 命令來解決此問題。要清...
    程式設計 發佈於2024-11-06
  • 如何加速 Matplotlib 繪圖以提高效能?
    如何加速 Matplotlib 繪圖以提高效能?
    為什麼 Matplotlib 這麼慢? 在評估 Python 繪圖庫時,考慮效能很重要。 Matplotlib 是一個廣泛使用的函式庫,它看起來可能很緩慢,引發了關於加快速度或探索替代選項的問題。讓我們深入研究這個問題並探索可能的解決方案。 提供的範例展示了具有多個子圖和資料更新的圖。使用 Matp...
    程式設計 發佈於2024-11-06
  • 使用畫布調整影像大小時如何克服鋸齒狀邊緣和模糊結果?
    使用畫布調整影像大小時如何克服鋸齒狀邊緣和模糊結果?
    解決在JavaScript 中使用Canvas 調整影像大小時的平滑問題在JavaScript 中使用Canvas 調整影像大小有時會導致明顯的鋸齒狀邊緣或模糊。為了實現平滑的調整大小,可以採用一種稱為向下步進的技術。 在大多數瀏覽器中,預設使用線性內插法來調整大小。雙三次插值可產生更平滑的結果,涉...
    程式設計 發佈於2024-11-06
  • 如何解決 MySQL C# 中的文字編碼問題?
    如何解決 MySQL C# 中的文字編碼問題?
    修復MySQL C# 中的文字編碼問題使用實體框架在C# 中處理MySQL 資料庫時,使用者可能會遇到文字編碼問題,特別是帶有特殊字符,例如“ë”,渲染不正確。本文探討了解決此常見問題的最合適的解決方案。 要修正編碼問題,必須執行以下操作:驗證排序規則設定: 確保所涉及的資料庫或表的排序規則與UTF...
    程式設計 發佈於2024-11-06
  • 如何將美麗搜尋與 Node.js 集成
    如何將美麗搜尋與 Node.js 集成
    作為 Node.js 開發人員,建立能夠提供快速且準確的搜尋結果的應用程式非常重要。使用者期望立即得到相關的回應,但實現起來可能具有挑戰性,特別是在處理大型資料集時。 這就是美麗搜尋的用武之地——一個為輕鬆滿足這些需求而構建的搜尋引擎。 什麼是美麗搜尋? Meilisearch ...
    程式設計 發佈於2024-11-06
  • 平行 JavaScript 機
    平行 JavaScript 機
    作者:Vladas Saulis,PE Prodata,克萊佩達,立陶宛 2024 年 5 月 18 日 抽象的 本文提出了一種新的程式設計模型,可以以簡單且自動平衡的方式利用多核心 CPU 系統。該模型還提出了一種更簡單的程式設計範例,用於在大多數大規模平行計算領域(例如天氣預報、核子物理、搜尋引...
    程式設計 發佈於2024-11-06

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3