Futures?
源代碼: Lib/asyncio/futures.py, Lib/asyncio/base_futures.py
Future 對(duì)象用來(lái)鏈接 底層回調(diào)式代碼 和高層異步/等待式代碼。
Future 函數(shù)?
- asyncio.isfuture(obj)?
如果 obj 為下面任意對(duì)象,返回
True
:一個(gè)
asyncio.Future
類(lèi)的實(shí)例,一個(gè)
asyncio.Task
類(lèi)的實(shí)例,帶有
_asyncio_future_blocking
屬性的類(lèi)似 Future 的對(duì)象。
3.5 新版功能.
- asyncio.ensure_future(obj, *, loop=None)?
返回:
obj 參數(shù)會(huì)是保持原樣,如果 obj 是
Future
、Task
或 類(lèi)似 Future 的對(duì)象(isfuture()
用于測(cè)試。)封裝了 obj 的
Task
對(duì)象,如果 obj 是一個(gè)協(xié)程 (使用iscoroutine()
進(jìn)行檢測(cè));在此情況下該協(xié)程將通過(guò)ensure_future()
加入執(zhí)行計(jì)劃。等待 obj 的
Task
對(duì)象,如果 obj 是一個(gè)可等待對(duì)象(inspect.isawaitable()
用于測(cè)試)
如果 obj 不是上述對(duì)象會(huì)引發(fā)一個(gè)
TypeError
異常。重要
查看
create_task()
函數(shù),它是創(chuàng)建新任務(wù)的首選途徑。Save a reference to the result of this function, to avoid a task disappearing mid execution.
在 3.5.1 版更改: 這個(gè)函數(shù)接受任意 awaitable 對(duì)象。
3.10 版后已移除: 如果 obj 不是 Future 類(lèi)對(duì)象同時(shí)未指定 loop 并且沒(méi)有正在運(yùn)行的事件循環(huán)則會(huì)發(fā)出棄用警告。
- asyncio.wrap_future(future, *, loop=None)?
將一個(gè)
concurrent.futures.Future
對(duì)象封裝到asyncio.Future
對(duì)象中。3.10 版后已移除: 如果 future 不是 Future 類(lèi)對(duì)象同時(shí)未指定 loop 并且沒(méi)有正在運(yùn)行的事件循環(huán)則會(huì)發(fā)出棄用警告。
Future 對(duì)象?
- class asyncio.Future(*, loop=None)?
一個(gè) Future 代表一個(gè)異步運(yùn)算的最終結(jié)果。線程不安全。
Future 是一個(gè) awaitable 對(duì)象。協(xié)程可以等待 Future 對(duì)象直到它們有結(jié)果或異常集合或被取消。
通常 Future 用于支持底層回調(diào)式代碼(例如在協(xié)議實(shí)現(xiàn)中使用asyncio transports) 與高層異步/等待式代碼交互。
經(jīng)驗(yàn)告訴我們永遠(yuǎn)不要面向用戶(hù)的接口暴露 Future 對(duì)象,同時(shí)建議使用
loop.create_future()
來(lái)創(chuàng)建 Future 對(duì)象。這種方法可以讓 Future 對(duì)象使用其它的事件循環(huán)實(shí)現(xiàn),它可以注入自己的優(yōu)化實(shí)現(xiàn)。在 3.7 版更改: 加入對(duì)
contextvars
模塊的支持。3.10 版后已移除: 如果未指定 loop 并且沒(méi)有正在運(yùn)行的事件循環(huán)則會(huì)發(fā)出棄用警告。
- result()?
返回 Future 的結(jié)果。
如果 Future 狀態(tài)為 完成 ,并由
set_result()
方法設(shè)置一個(gè)結(jié)果,則返回這個(gè)結(jié)果。如果 Future 狀態(tài)為 完成 ,并由
set_exception()
方法設(shè)置一個(gè)異常,那么這個(gè)方法會(huì)引發(fā)異常。如果 Future 已 取消,方法會(huì)引發(fā)一個(gè)
CancelledError
異常。如果 Future 的結(jié)果還不可用,此方法會(huì)引發(fā)一個(gè)
InvalidStateError
異常。
- set_result(result)?
將 Future 標(biāo)記為 完成 并設(shè)置結(jié)果。
如果 Future 已經(jīng) 完成 則拋出一個(gè)
InvalidStateError
錯(cuò)誤。
- set_exception(exception)?
將 Future 標(biāo)記為 完成 并設(shè)置一個(gè)異常。
如果 Future 已經(jīng) 完成 則拋出一個(gè)
InvalidStateError
錯(cuò)誤。
- done()?
如果 Future 為已 完成 則返回
True
。如果 Future 為 取消 或調(diào)用
set_result()
設(shè)置了結(jié)果或調(diào)用set_exception()
設(shè)置了異常,那么它就是 完成 。
- cancelled()?
如果 Future 已 取消 則返回
True
這個(gè)方法通常在設(shè)置結(jié)果或異常前用來(lái)檢查 Future 是否已 取消 。
if not fut.cancelled(): fut.set_result(42)
- add_done_callback(callback, *, context=None)?
添加一個(gè)在 Future 完成 時(shí)運(yùn)行的回調(diào)函數(shù)。
調(diào)用 callback 時(shí),F(xiàn)uture 對(duì)象是它的唯一參數(shù)。
如果調(diào)用這個(gè)方法時(shí) Future 已經(jīng) 完成,回調(diào)函數(shù)會(huì)被
loop.call_soon()
調(diào)度。可選鍵值類(lèi)的參數(shù) context 允許 callback 運(yùn)行在一個(gè)指定的自定義
contextvars.Context
對(duì)象中。如果沒(méi)有提供 context ,則使用當(dāng)前上下文。可以用
functools.partial()
給回調(diào)函數(shù)傳遞參數(shù),例如:# Call 'print("Future:", fut)' when "fut" is done. fut.add_done_callback( functools.partial(print, "Future:"))
在 3.7 版更改: 加入鍵值類(lèi)形參 context。請(qǐng)參閱 PEP 567 查看更多細(xì)節(jié)。
- remove_done_callback(callback)?
從回調(diào)列表中移除 callback 。
返回被移除的回調(diào)函數(shù)的數(shù)量,通常為1,除非一個(gè)回調(diào)函數(shù)被添加多次。
- cancel(msg=None)?
取消 Future 并調(diào)度回調(diào)函數(shù)。
如果 Future 已經(jīng) 完成 或 取消 ,返回
False
。否則將 Future 狀態(tài)改為 取消 并在調(diào)度回調(diào)函數(shù)后返回True
。在 3.9 版更改: Added the msg parameter.
Deprecated since version 3.11, will be removed in version 3.14: msg parameter is ambiguous when multiple
cancel()
are called with different cancellation messages. The argument will be removed.
- exception()?
返回 Future 已設(shè)置的異常。
只有 Future 在 完成 時(shí)才返回異常(或者
None
,如果沒(méi)有設(shè)置異常)。如果 Future 已 取消,方法會(huì)引發(fā)一個(gè)
CancelledError
異常。如果 Future 還沒(méi) 完成 ,這個(gè)方法會(huì)引發(fā)一個(gè)
InvalidStateError
異常。
- get_loop()?
返回 Future 對(duì)象已綁定的事件循環(huán)。
3.7 新版功能.
這個(gè)例子創(chuàng)建一個(gè) Future 對(duì)象,創(chuàng)建和調(diào)度一個(gè)異步任務(wù)去設(shè)置 Future 結(jié)果,然后等待其結(jié)果:
async def set_after(fut, delay, value):
# Sleep for *delay* seconds.
await asyncio.sleep(delay)
# Set *value* as a result of *fut* Future.
fut.set_result(value)
async def main():
# Get the current event loop.
loop = asyncio.get_running_loop()
# Create a new Future object.
fut = loop.create_future()
# Run "set_after()" coroutine in a parallel Task.
# We are using the low-level "loop.create_task()" API here because
# we already have a reference to the event loop at hand.
# Otherwise we could have just used "asyncio.create_task()".
loop.create_task(
set_after(fut, 1, '... world'))
print('hello ...')
# Wait until *fut* has a result (1 second) and print it.
print(await fut)
asyncio.run(main())
重要
該 Future 對(duì)象是為了模仿 concurrent.futures.Future
類(lèi)。主要差異包含:
與 asyncio 的 Future 不同,
concurrent.futures.Future
實(shí)例不是可等待對(duì)象。asyncio.Future.result()
和asyncio.Future.exception()
不接受 timeout 參數(shù)。Future 沒(méi)有 完成 時(shí)
asyncio.Future.result()
和asyncio.Future.exception()
拋出一個(gè)InvalidStateError
異常。使用
asyncio.Future.add_done_callback()
注冊(cè)的回調(diào)函數(shù)不會(huì)立即調(diào)用,而是被loop.call_soon()
調(diào)度。asyncio Future 不能兼容
concurrent.futures.wait()
和concurrent.futures.as_completed()
函數(shù)。asyncio.Future.cancel()
接受一個(gè)可選的msg
參數(shù),但concurrent.futures.cancel()
無(wú)此參數(shù)。Deprecated since version 3.11, will be removed in version 3.14: msg parameter is ambiguous when multiple
cancel()
are called with different cancellation messages. The argument will be removed.