Python異步之怎么保護(hù)任務(wù)免于取消
本文講解"Python異步之如何保護(hù)任務(wù)免于取消",希望能夠解決相關(guān)問題。
Asyncio 任務(wù)可以通過調(diào)用它們的 cancel() 方法來取消。我們可以通過將任務(wù)包裝在對 asyncio.shield() 的調(diào)用中來保護(hù)任務(wù)不被取消。
1. 什么是 Asyncio shield()
asyncio.shield() 函數(shù)在 Future 中包裝了一個可等待對象,它將吸收要取消的請求。
這意味著被屏蔽的未來可以傳遞給可能嘗試取消它的任務(wù),并且取消請求看起來像是成功的,除了被屏蔽的任務(wù)或協(xié)程將繼續(xù)運行。
它可能在 asyncio 程序中很有用,其中某些任務(wù)可以取消,但其他任務(wù)(可能具有更高優(yōu)先級)則不能。
它也可能在某些任務(wù)可以安全取消的程序中很有用,例如那些在設(shè)計時考慮了 asyncio 的任務(wù),而其他任務(wù)則不能安全終止,因此必須避免取消。
現(xiàn)在我們知道了 asyncio.shield() 是什么,讓我們看看如何使用它。
2. 如何使用 Asyncio shield()
asyncio.shield() 函數(shù)將保護(hù)另一個任務(wù)或協(xié)程不被取消。它以一個可等待對象作為參數(shù)并返回一個 asyncio.Future 對象。
然后可以直接等待 Future 對象或?qū)⑵鋫鬟f給另一個任務(wù)或協(xié)程。
... # shield a task from cancellation shielded = asyncio.shield(task) # await the shielded task await shielded
返回的 Future 可以通過調(diào)用 cancel() 方法取消。
如果內(nèi)部任務(wù)正在運行,請求將被報告為成功。
... # cancel a shielded task was_canceld = shielded.cancel()
任何等待 Future 對象的協(xié)程都會引發(fā) asyncio.CancelledError,這可能需要處理。
... try: # await the shielded task await asyncio.shield(task) except asyncio.CancelledError: # ...
重要的是,對 Future 對象的取消請求不會傳播到內(nèi)部任務(wù)。這意味著取消請求被護(hù)盾吸收了。
... # create a task task = asyncio.create_task(coro()) # create a shield shield = asyncio.shield(task) # cancel the shield (does not cancel the task) shield.cancel()
如果協(xié)程被提供給 asyncio.shield() 函數(shù),它將被包裝在 asyncio.Task() 中并立即調(diào)度。
這意味著不需要等待屏蔽來讓內(nèi)部協(xié)程運行。
如果被屏蔽的任務(wù)被取消,取消請求將向上傳播到屏蔽,屏蔽也將被取消。
... # create a task task = asyncio.create_task(coro()) # create a shield shield = asyncio.shield(task) # cancel the task (also cancels the shield) task.cancel()
現(xiàn)在我們知道如何使用 asyncio.shield() 函數(shù),讓我們看一些有效的例子。
3. 示例
我們可以探索如何使用 asyncio.shield() 來保護(hù)任務(wù)不被取消。
在這個例子中,我們定義了一個簡單的協(xié)程任務(wù),它接受一個整數(shù)參數(shù),休眠一秒鐘,然后返回參數(shù)。然后可以創(chuàng)建協(xié)程并將其安排為任務(wù)。
我們可以定義第二個協(xié)程,它接受一個任務(wù),休眠幾分之一秒,然后取消提供的任務(wù)。
在主協(xié)程中,我們可以屏蔽第一個任務(wù),然后將其傳遞給第二個任務(wù),然后等待被屏蔽的任務(wù)。
期望是屏蔽將被取消并保持內(nèi)部任務(wù)完好無損。取消將中斷主協(xié)程。我們可以在程序結(jié)束時檢查內(nèi)部任務(wù)的狀態(tài),我們希望它已經(jīng)正常完成,而不管屏蔽上的取消請求如何。
# SuperFastPython.com # example of using asyncio shield to protect a task from cancellation import asyncio # define a simple asynchronous async def simple_task(number): # block for a moment await asyncio.sleep(1) # return the argument return number # cancel the given task after a moment async def cancel_task(task): # block for a moment await asyncio.sleep(0.2) # cancel the task was_cancelled = task.cancel() print(f'cancelled: {was_cancelled}') # define a simple coroutine async def main(): # create the coroutine coro = simple_task(1) # create a task task = asyncio.create_task(coro) # created the shielded task shielded = asyncio.shield(task) # create the task to cancel the previous task asyncio.create_task(cancel_task(shielded)) # handle cancellation try: # await the shielded task result = await shielded # report the result print(f'>got: {result}') except asyncio.CancelledError: print('shielded was cancelled') # wait a moment await asyncio.sleep(1) # report the details of the tasks print(f'shielded: {shielded}') print(f'task: {task}') # start asyncio.run(main())
運行示例首先創(chuàng)建 main() 協(xié)程并將其用作應(yīng)用程序的入口點。創(chuàng)建任務(wù)協(xié)程,然后將其包裝并安排在任務(wù)中。然后該任務(wù)就不會被取消。
然后將屏蔽的任務(wù)傳遞給 cancel_task() 協(xié)程,該協(xié)程包裝在任務(wù)中并進(jìn)行調(diào)度。主協(xié)程然后等待受保護(hù)的任務(wù),該任務(wù)需要 CancelledError 異常。
任務(wù)運行片刻然后休眠。取消任務(wù)運行片刻,休眠,恢復(fù)然后取消屏蔽任務(wù)。取消請求報告它已成功。
這會在受保護(hù)的 Future 中引發(fā) CancelledError 異常,但不會在內(nèi)部任務(wù)中引發(fā)。
main() 協(xié)程恢復(fù)并響應(yīng) CancelledError 異常,報告一條消息。然后它會睡一會兒。任務(wù)恢復(fù)、完成并返回一個值。
最后,main() 協(xié)程恢復(fù),并報告被屏蔽的未來和內(nèi)部任務(wù)的狀態(tài)。我們可以看到屏蔽的未來被標(biāo)記為已取消,而內(nèi)部任務(wù)被標(biāo)記為正常完成并提供返回值。
此示例突出顯示了如何使用防護(hù)罩來成功保護(hù)內(nèi)部任務(wù)不被取消。
cancelled: True shielded was cancelled shielded: <future cancelled> task: <task?finished?name='task-2'?coro=
- Python中find函數(shù)如何使用
- Python異步怎么使用等待有時間限制協(xié)程
- Python異步之在Asyncio中怎么運行阻塞任務(wù)
- Python反射機(jī)制怎么應(yīng)用
- Python異步之上下文管理器怎么使用
- Python異步之生成器怎么使用
- Python異步之如何獲取當(dāng)前和正在運行任務(wù)
- Python中Matplotlib圖像如何添加標(biāo)簽
- 如何封裝Python時間處理庫創(chuàng)建自己的TimeUtil類
- Python HTTP標(biāo)頭
- Python HTTP數(shù)據(jù)下載
- Python 網(wǎng)絡(luò)接口
- Python SMTP
- Python POP3
- Python 上傳數(shù)據(jù)
- Python 代理服務(wù)器
- Python 遠(yuǎn)程過程調(diào)用
- Python 線程并發(fā)
- Python 同步線程
- Python 反應(yīng)式編程