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ì)是保持原樣,如果 objFuture、 Task 或 類(lèi)似 Future 的對(duì)象( isfuture() 用于測(cè)試。)

  • 封裝了 objTask 對(duì)象,如果 obj 是一個(gè)協(xié)程 (使用 iscoroutine() 進(jìn)行檢測(cè));在此情況下該協(xié)程將通過(guò) ensure_future() 加入執(zhí)行計(jì)劃。

  • 等待 objTask 對(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)。主要差異包含: