try:
loop.run_forever()
finally:
loop.run_until_complete(loop.shutdown_asyncgens())
loop.close()
3.6 新版功能.
coroutine loop.shutdown_default_executor()
Schedule the closure of the default executor and wait for it to join all of
the threads in the ThreadPoolExecutor
.
Once this method has been called,
using the default executor with loop.run_in_executor()
will raise a RuntimeError
.
Do not call this method when using asyncio.run()
,
as the latter handles default executor shutdown automatically.
3.9 新版功能.
loop.call_soon(callback, *args, context=None)
安排 callback callback 在事件循环的下一次迭代时附带 args 参数被调用。
Return an instance of asyncio.Handle
,
which can be used later to cancel the callback.
回调按其注册顺序被调用。每个回调仅被调用一次。
The optional keyword-only context argument specifies a
custom contextvars.Context
for the callback to run in.
Callbacks use the current context when no context is provided.
Unlike call_soon_threadsafe()
, this method is not thread-safe.
loop.call_soon_threadsafe(callback, *args, context=None)
A thread-safe variant of call_soon()
. When scheduling callbacks from
another thread, this function must be used, since call_soon()
is not
thread-safe.
如果在已被关闭的循环上调用则会引发 RuntimeError
。 这可能会在主应用程序被关闭时在二级线程上发生。
参见 concurrency and multithreading 部分的文档。
在 3.7 版更改: 加入键值类形参 context。请参阅 PEP 567 查看更多细节。
大多数 asyncio
的调度函数不让传递关键字参数。为此,请使用 functools.partial()
:
# will schedule "print("Hello", flush=True)"
loop.call_soon(
functools.partial(print, "Hello", flush=True))
使用 partial 对象通常比使用lambda更方便,asyncio 在调试和错误消息中能更好的呈现 partial 对象。
事件循环提供安排调度函数在将来某个时刻调用的机制。事件循环使用单调时钟来跟踪时间。
loop.call_later(delay, callback, *args, context=None)
安排 callback 在给定的 delay 秒(可以是 int 或者 float)后被调用。
返回一个 asyncio.TimerHandle
实例,该实例能用于取消回调。
callback 只被调用一次。如果两个回调被安排在同样的时间点,执行顺序未限定。
可选的位置参数 args 在被调用的时候传递给 callback 。 如果你想把关键字参数传递给 callback ,请使用 functools.partial()
。
可选键值类的参数 context 允许 callback 运行在一个指定的自定义 contextvars.Context
对象中。如果没有提供 context ,则使用当前上下文。
在 3.7 版更改: 加入键值类形参 context。请参阅 PEP 567 查看更多细节。
在 3.8 版更改: 在 Python 3.7 和更早版本的默认事件循环实现中, delay 不能超过一天。 这在 Python 3.8 中已被修复。
loop.call_at(when, callback, *args, context=None)
安排 callback 在给定的绝对时间戳 when (int 或 float) 被调用,使用与 loop.time()
同样的时间参考。
本方法的行为和 call_later()
方法相同。
返回一个 asyncio.TimerHandle
实例,该实例能用于取消回调。
在 3.7 版更改: 加入键值类形参 context。请参阅 PEP 567 查看更多细节。
在 3.8 版更改: 在 Python 3.7 和更早版本的默认事件循环实现中,when 和当前时间相差不能超过一天。 在这 Python 3.8 中已被修复。
loop.create_future()
创建一个附加到事件循环中的 asyncio.Future
对象。
这是在asyncio中创建Futures的首选方式。这让第三方事件循环可以提供Future 对象的替代实现(更好的性能或者功能)。
3.5.2 新版功能.
loop.create_task(coro, *, name=None, context=None)
Schedule the execution of coroutine coro.
Return a Task
object.
第三方的事件循环可以使用它们自己的 Task
子类来满足互操作性。这种情况下结果类型是一个 Task
的子类。
如果提供了 name 参数且不为 None
,它会使用 Task.set_name()
来设为任务的名称。
An optional keyword-only context argument allows specifying a
custom contextvars.Context
for the coro to run in.
The current context copy is created when no context is provided.
在 3.8 版更改: 添加了 name 参数。
在 3.11 版更改: Added the context parameter.
loop.set_task_factory(factory)
设置一个任务工厂,它将由 loop.create_task()
来使用。
If factory is None
the default task factory will be set.
Otherwise, factory must be a callable with the signature matching
(loop, coro, context=None)
, where loop is a reference to the active
event loop, and coro is a coroutine object. The callable
must return a asyncio.Future
-compatible object.
coroutine loop.create_connection(protocol_factory, host=None, port=None, *, ssl=None, family=0, proto=0, flags=0, sock=None, local_addr=None, server_hostname=None, ssl_handshake_timeout=None, ssl_shutdown_timeout=None, happy_eyeballs_delay=None, interleave=None)
打开一个流式传输连接,连接到由 host 和 port 指定的地址。
The socket family can be either AF_INET
or
AF_INET6
depending on host (or the family
argument, if provided).
The socket type will be SOCK_STREAM
.
protocol_factory 必须为一个返回 asyncio 协议 实现的可调用对象。
这个方法会尝试在后台创建连接。当创建成功,返回 (transport, protocol)
组合。
底层操作的大致的执行顺序是这样的:
创建连接并为其创建一个 传输。
不带参数地调用 protocol_factory 并预期返回一个 协议 实例。
协议实例通过调用其 connection_made()
方法与传输进行配对。
成功时返回一个 (transport, protocol)
元组。
创建的传输是一个具体实现相关的双向流。
其他参数:
ssl: 如果给定该参数且不为假值,则会创建一个 SSL/TLS 传输(默认创建一个纯 TCP 传输)。 如果 ssl 是一个 ssl.SSLContext
对象,则会使用此上下文来创建传输对象;如果 ssl 为 True
,则会使用从 ssl.create_default_context()
返回的默认上下文。
SSL/TLS security considerations
server_hostname 设置或重载目标服务器的证书将要匹配的主机名。 应当只在 ssl 不为 None
时传入。 默认情况下会使用 host 参数的值。 如果 host 为空那就没有默认值,你必须为 server_hostname 传入一个值。 如果 server_hostname 为空字符串,则主机名匹配会被禁用(这是一个严重的安全风险,使得潜在的中间人攻击成为可能)。
family, proto, flags 是可选的地址族、协议和标志,它们会被传递给 getaddrinfo() 来对 host 进行解析。如果要指定的话,这些都应该是来自于 socket
模块的对应常量。
如果给出 happy_eyeballs_delay,它将为此链接启用 Happy Eyeballs。 该函数应当为一个表示在开始下一个并行尝试之前要等待连接尝试完成的秒数的浮点数。 这也就是在 RFC 8305 中定义的 "连接尝试延迟"。 该 RFC 所推荐的合理默认值为 0.25
(250 毫秒)。
interleave 控制当主机名解析为多个 IP 地址时的地址重排序。 如果该参数为 0
或未指定,则不会进行重排序,这些地址会按 getaddrinfo()
所返回的顺序进行尝试。 如果指定了一个正整数,这些地址会按地址族交错排列,而指定的整数会被解读为 RFC 8305 所定义的 "首个地址族计数"。 如果 happy_eyeballs_delay 未指定则默认值为 0
,否则为 1
。
如果给出 sock,它应当是一个已存在、已连接并将被传输所使用的 socket.socket
对象。 如果给出了 sock,则 host, port, family, proto, flags, happy_eyeballs_delay, interleave 和 local_addr 都不应当被指定。
The sock argument transfers ownership of the socket to the
transport created. To close the socket, call the transport's
close()
method.
如果给出 local_addr,它应当是一个用来在本地绑定套接字的 (local_host, local_port)
元组。 local_host 和 local_port 会使用 getaddrinfo()
来查找,这与 host 和 port 类似。
ssl_handshake_timeout 是(用于 TLS 连接的)在放弃连接之前要等待 TLS 握手完成的秒数。 如果参数为 None
则使用 (默认的) 60.0
。
ssl_shutdown_timeout is the time in seconds to wait for the SSL shutdown
to complete before aborting the connection. 30.0
seconds if None
(default).
在 3.5 版更改: ProactorEventLoop
类中添加 SSL/TLS 支持。
在 3.6 版更改: The socket option TCP_NODELAY
is set by default
for all TCP connections.
在 3.7 版更改: 添加了 ssl_handshake_timeout 参数。
在 3.8 版更改: 增加了 happy_eyeballs_delay 和 interleave 形参。
Happy Eyeballs Algorithm: Success with Dual-Stack Hosts.
When a server's IPv4 path and protocol are working, but the server's
IPv6 path and protocol are not working, a dual-stack client
application experiences significant connection delay compared to an
IPv4-only client. This is undesirable because it causes the
dual-stack client to have a worse user experience. This document
specifies requirements for algorithms that reduce this user-visible
delay and provides an algorithm.
For more information: https://datatracker.ietf.org/doc/html/rfc6555
在 3.11 版更改: Added the ssl_shutdown_timeout parameter.
open_connection()
函数是一个高层级的替代 API。 它返回一对 (StreamReader
, StreamWriter
),可在 async/await 代码中直接使用。
coroutine loop.create_datagram_endpoint(protocol_factory, local_addr=None, remote_addr=None, *, family=0, proto=0, flags=0, reuse_port=None, allow_broadcast=None, sock=None)
创建一个数据报连接。
The socket family can be either AF_INET
,
AF_INET6
, or AF_UNIX
,
depending on host (or the family argument, if provided).
The socket type will be SOCK_DGRAM
.
protocol_factory 必须为一个返回 协议 实现的可调用对象。
成功时返回一个 (transport, protocol)
元组。
其他参数:
如果给出 local_addr,它应当是一个用来在本地绑定套接字的 (local_host, local_port)
元组。 local_host 和 local_port 是使用 getaddrinfo()
来查找的。
remote_addr,如果指定的话,就是一个 (remote_host, remote_port)
元组,用于同一个远程地址连接。remote_host 和 remote_port 是使用 getaddrinfo()
来查找的。
family, proto, flags 是可选的地址族,协议和标志,其会被传递给 getaddrinfo()
来完成 host 的解析。如果要指定的话,这些都应该是来自于 socket
模块的对应常量。
reuse_port tells the kernel to allow this endpoint to be bound to the
same port as other existing endpoints are bound to, so long as they all
set this flag when being created. This option is not supported on Windows
and some Unixes. If the SO_REUSEPORT
constant is not
defined then this capability is unsupported.
allow_broadcast 告知内核允许此端点向广播地址发送消息。
sock 可选择通过指定此值用于使用一个预先存在的,已经处于连接状态的 socket.socket
对象,并将其提供给此传输对象使用。如果指定了这个值, local_addr 和 remote_addr 就应该被忽略 (必须为 None
)。
The sock argument transfers ownership of the socket to the
transport created. To close the socket, call the transport's
close()
method.
参见 UDP echo 客户端协议 和 UDP echo 服务端协议 的例子。
在 3.4.4 版更改: The family, proto, flags, reuse_address, reuse_port,
allow_broadcast, and sock parameters were added.
在 3.8.1 版更改: The reuse_address parameter is no longer supported, as using
SO_REUSEADDR
poses a significant security concern for
UDP. Explicitly passing reuse_address=True
will raise an exception.
当具有不同 UID 的多个进程将套接字赋给具有 SO_REUSEADDR
的相同 UDP 套接字地址时,传入的数据包可能会在套接字间随机分配。
For supported platforms, reuse_port can be used as a replacement for
similar functionality. With reuse_port,
SO_REUSEPORT
is used instead, which specifically
prevents processes with differing UIDs from assigning sockets to the same
socket address.
在 3.8 版更改: 添加WIndows的支持。
在 3.11 版更改: The reuse_address parameter, disabled since Python 3.9.0, 3.8.1,
3.7.6 and 3.6.10, has been entirely removed.
coroutine loop.create_unix_connection(protocol_factory, path=None, *, ssl=None, sock=None, server_hostname=None, ssl_handshake_timeout=None, ssl_shutdown_timeout=None)
创建Unix 连接
The socket family will be AF_UNIX
; socket
type will be SOCK_STREAM
.
成功时返回一个 (transport, protocol)
元组。
path 是所要求的 Unix 域套接字的名字,除非指定了 sock 形参。 抽象的 Unix 套接字, str
, bytes
和 Path
路径都是受支持的。
请查看 loop.create_connection()
方法的文档了解有关此方法的参数的信息。
可用性: Unix。
在 3.7 版更改: 增加了 ssl_handshake_timeout 参数。现在 path 参数可以是一个 path-like object 。
在 3.11 版更改: Added the ssl_shutdown_timeout parameter.
coroutine loop.create_server(protocol_factory, host=None, port=None, *, family=socket.AF_UNSPEC, flags=socket.AI_PASSIVE, sock=None, backlog=100, ssl=None, reuse_address=
None, reuse_port=None, ssl_handshake_timeout=None, ssl_shutdown_timeout=None, start_serving=True)
Create a TCP server (socket type SOCK_STREAM
) listening
on port of the host address.
返回一个 Server
对象。
protocol_factory 必须为一个返回 协议 实现的可调用对象。
host 形参可被设为几种类型,它确定了服务器所应监听的位置:
如果 host 是一个字符串,则 TCP 服务器会被绑定到 host 所指明的单一网络接口。
如果 host 是一个字符串序列,则 TCP 服务器会被绑定到序列所指明的所有网络接口。
如果 host 是一个空字符串或 None
,则会应用所有接口并将返回包含多个套接字的列表(通常是一个 IPv4 的加一个 IPv6 的)。
可以设置 port 参数来指定服务器应该监听哪个端口。如果为 0
或者 None
(默认),将选择一个随机的未使用的端口(注意,如果 host 解析到多个网络接口,将为每个接口选择一个不同的随机端口)。
family can be set to either socket.AF_INET
or
AF_INET6
to force the socket to use IPv4 or IPv6.
If not set, the family will be determined from host name
(defaults to AF_UNSPEC
).
flags 是用于 getaddrinfo()
的位掩码。
可以选择指定 sock 以便使用预先存在的套接字对象。 如果指定了此参数,则不可再指定 host 和 port。
The sock argument transfers ownership of the socket to the
server created. To close the socket, call the server's
close()
method.
backlog 是传递给 listen()
的最大排队连接的数量(默认为100)。
ssl 可被设置为一个 SSLContext
实例以在所接受的连接上启用 TLS。
reuse_address 告知内核要重用一个处于 TIME_WAIT
状态的本地套接字,而不是等待其自然超时失效。 如果未指定此参数则在 Unix 上将自动设置为 True
。
reuse_port 告知内核,只要在创建的时候都设置了这个标志,就允许此端点绑定到其它端点列表所绑定的同样的端口上。这个选项在 Windows 上是不支持的。
ssl_handshake_timeout 是(用于 TLS 服务器的)在放弃连接之前要等待 TLS 握手完成的秒数。 如果参数为 (默认值) None
则为 60.0
秒。
ssl_shutdown_timeout is the time in seconds to wait for the SSL shutdown
to complete before aborting the connection. 30.0
seconds if None
(default).
start_serving 设置成 True
(默认值) 会导致创建server并立即开始接受连接。设置成 False
,用户需要等待 Server.start_serving()
或者 Server.serve_forever()
以使server开始接受连接。
在 3.5 版更改: ProactorEventLoop
类中添加 SSL/TLS 支持。
在 3.5.1 版更改: host 形参可以是一个字符串的序列。
在 3.6 版更改: Added ssl_handshake_timeout and start_serving parameters.
The socket option TCP_NODELAY
is set by default
for all TCP connections.
在 3.11 版更改: Added the ssl_shutdown_timeout parameter.
start_server()
函数是一个高层级的替代 API,它返回一对 StreamReader
和 StreamWriter
,可在 async/await 代码中使用。
coroutine loop.create_unix_server(protocol_factory, path=None, *, sock=None, backlog=100, ssl=None, ssl_handshake_timeout=None, ssl_shutdown_timeout=None, start_serving=True)
Similar to loop.create_server()
but works with the
AF_UNIX
socket family.
path 是必要的 Unix 域套接字名称,除非提供了 sock 参数。 抽象的 Unix 套接字, str
, bytes
和 Path
路径都是受支持的。
请查看 loop.create_server()
方法的文档了解有关此方法的参数的信息。
可用性: Unix。
在 3.7 版更改: 增加了 ssl_handshake_timeout 和 start_serving 参数。现在 path 参数可以是一个 Path
对象。
在 3.11 版更改: Added the ssl_shutdown_timeout parameter.
coroutine loop.connect_accepted_socket(protocol_factory, sock, *, ssl=None, ssl_handshake_timeout=None, ssl_shutdown_timeout=None)
将已被接受的连接包装成一个传输/协议对。
此方法可被服务器用来接受 asyncio 以外的连接,但是使用 asyncio 来处理它们。
protocol_factory 必须为一个返回 协议 实现的可调用对象。
sock 是一个预先存在的套接字对象,它是由 socket.accept
返回的。
The sock argument transfers ownership of the socket to the
transport created. To close the socket, call the transport's
close()
method.
ssl 可被设置为一个 SSLContext
以在接受的连接上启用 SSL。
ssl_handshake_timeout 是(为一个SSL连接)在中止连接前,等待SSL握手完成的时间【单位秒】。如果为 None
(缺省) 则是 60.0
秒。
ssl_shutdown_timeout is the time in seconds to wait for the SSL shutdown
to complete before aborting the connection. 30.0
seconds if None
(default).
返回一个 (transport, protocol)
对。
3.5.3 新版功能.
在 3.7 版更改: 添加了 ssl_handshake_timeout 参数。
在 3.11 版更改: Added the ssl_shutdown_timeout parameter.
coroutine loop.sendfile(transport, file, offset=0, count=None, *, fallback=True)
将 file 通过 transport 发送。 返回所发送的字节总数。
如果可用的话,该方法将使用高性能的 os.sendfile()
。
file 必须是个二进制模式打开的常规文件对象。
offset 指明从何处开始读取文件。 如果指定了 count,它是要传输的字节总数而不再一直发送文件直至抵达 EOF。 文件位置总是会被更新,即使此方法引发了错误,并可以使用 file.tell()
来获取实际发送的字节总数。
fallback 设为 True
会使得 asyncio 在平台不支持 sendfile 系统调用时手动读取并发送文件(例如 Windows 或 Unix 上的 SSL 套接字)。
如果系统不支持 sendfile 系统调用且 fallback 为 False
则会引发 SendfileNotAvailableError
。
3.7 新版功能.
coroutine loop.start_tls(transport, protocol, sslcontext, *, server_side=False, server_hostname=None, ssl_handshake_timeout=None, ssl_shutdown_timeout=None)
将现有基于传输的连接升级到 TLS。
Create a TLS coder/decoder instance and insert it between the transport
and the protocol. The coder/decoder implements both transport-facing
protocol and protocol-facing transport.
Return the created two-interface instance. After await, the protocol
must stop using the original transport and communicate with the returned
object only because the coder caches protocol-side data and sporadically
exchanges extra TLS session packets with transport.
In some situations (e.g. when the passed transport is already closing) this
may return None
.
transport 和 protocol 实例的方法与 create_server()
和 create_connection()
所返回的类似。
sslcontext :一个已经配置好的 SSLContext
实例。
当服务端连接已升级时 (如 create_server()
所创建的对象) server_side 会传入 True
。
server_hostname :设置或者覆盖目标服务器证书中相对应的主机名。
ssl_handshake_timeout 是(用于 TLS 连接的)在放弃连接之前要等待 TLS 握手完成的秒数。 如果参数为 None
则使用 (默认的) 60.0
。
ssl_shutdown_timeout is the time in seconds to wait for the SSL shutdown
to complete before aborting the connection. 30.0
seconds if None
(default).
3.7 新版功能.
在 3.11 版更改: Added the ssl_shutdown_timeout parameter.
loop.remove_reader(fd)
Stop monitoring the fd file descriptor for read availability. Returns
True
if fd was previously being monitored for reads.
loop.add_writer(fd, callback, *args)
开始监视 fd 文件描述符的写入可用性,一旦 fd 可用于写入,使用指定的参数调用 callback 。
使用 functools.partial()
传递关键字参数 给 callback.
loop.remove_writer(fd)
Stop monitoring the fd file descriptor for write availability. Returns
True
if fd was previously being monitored for writes.
另请查看 平台支持 一节了解以上方法的某些限制。
通常,使用基于传输的 API 的协议实现,例如 loop.create_connection()
和 loop.create_server()
比直接使用套接字的实现更快。 但是,在某些应用场景下性能并不非常重要,直接使用 socket
对象会更方便。
coroutine loop.sock_recv(sock, nbytes)
从 sock 接收至多 nbytes。 socket.recv()
的异步版本。
返回接收到的数据【bytes对象类型】。
sock 必须是个非阻塞socket。
在 3.7 版更改: 虽然这个方法总是被记录为协程方法,但它在 Python 3.7 之前的发行版中会返回一个 Future
。 从 Python 3.7 开始它则是一个 async def
方法。
coroutine loop.sock_recv_into(sock, buf)
从 sock 接收数据放入 buf 缓冲区。 模仿了阻塞型的 socket.recv_into()
方法。
返回写入缓冲区的字节数。
sock 必须是个非阻塞socket。
3.7 新版功能.
coroutine loop.sock_recvfrom(sock, bufsize)
Receive a datagram of up to bufsize from sock. Asynchronous version of
socket.recvfrom()
.
Return a tuple of (received data, remote address).
sock 必须是个非阻塞socket。
3.11 新版功能.
coroutine loop.sock_recvfrom_into(sock, buf, nbytes=0)
Receive a datagram of up to nbytes from sock into buf.
Asynchronous version of
socket.recvfrom_into()
.
Return a tuple of (number of bytes received, remote address).
sock 必须是个非阻塞socket。
3.11 新版功能.
coroutine loop.sock_sendall(sock, data)
将 data 发送到 sock 套接字。 socket.sendall()
的异步版本。
此方法会持续发送数据到套接字直至 data 中的所有数据发送完毕或是有错误发生。 当成功时会返回 None
。 当发生错误时,会引发一个异常。 此外,没有办法能确定有多少数据或是否有数据被连接的接收方成功处理。
sock 必须是个非阻塞socket。
在 3.7 版更改: 虽然这个方法一直被标记为协程方法。但是,Python 3.7 之前,该方法返回 Future
,从Python 3.7 开始,这个方法是 async def
方法。
coroutine loop.sock_sendto(sock, data, address)
Send a datagram from sock to address.
Asynchronous version of
socket.sendto()
.
Return the number of bytes sent.
sock 必须是个非阻塞socket。
3.11 新版功能.
coroutine loop.sock_connect(sock, address)
将 sock 连接到位于 address 的远程套接字。
socket.connect()
的异步版本。
sock 必须是个非阻塞socket。
在 3.5.2 版更改: address
不再需要被解析。 sock_connect
将尝试检查 address 是否已通过调用 socket.inet_pton()
被解析。 如果没有,则将使用 loop.getaddrinfo()
来解析 address。
loop.create_connection()
和 asyncio.open_connection()
。
coroutine loop.sock_accept(sock)
接受一个连接。 模仿了阻塞型的 socket.accept()
方法。
此 scoket 必须绑定到一个地址上并且监听连接。返回值是一个 (conn, address)
对,其中 conn 是一个 新*的套接字对象,用于在此连接上收发数据,*address 是连接的另一端的套接字所绑定的地址。
sock 必须是个非阻塞socket。
在 3.7 版更改: 虽然这个方法一直被标记为协程方法。但是,Python 3.7 之前,该方法返回 Future
,从Python 3.7 开始,这个方法是 async def
方法。
loop.create_server()
和 start_server()
。
coroutine loop.sock_sendfile(sock, file, offset=0, count=None, *, fallback=True)
在可能的情况下使用高性能的 os.sendfile
发送文件。 返回所发送的字节总数。
socket.sendfile()
的异步版本。
sock 必须为非阻塞型的 socket.SOCK_STREAM
socket
。
file 必须是个用二进制方式打开的常规文件对象。
offset 指明从何处开始读取文件。 如果指定了 count,它是要传输的字节总数而不再一直发送文件直至抵达 EOF。 文件位置总是会被更新,即使此方法引发了错误,并可以使用 file.tell()
来获取实际发送的字节总数。
当 fallback 被设为 True
时,会使用 asyncio 在平台不支持 sendfile 系统调用时手动读取并发送文件(例如 Windows 或 Unix 上的 SSL 套接字)。
如果系统不支持 sendfile 并且 fallback 为 False
,引发 SendfileNotAvailableError
异常。
sock 必须是个非阻塞socket。
3.7 新版功能.
coroutine loop.getaddrinfo(host, port, *, family=0, type=0, proto=0, flags=0)
异步版的 socket.getaddrinfo()
。
coroutine loop.connect_read_pipe(protocol_factory, pipe)
在事件循环中注册 pipe 的读取端。
protocol_factory 必须为一个返回 asyncio 协议 实现的可调用对象。
pipe 是个 类似文件型对象.
返回一对 (transport, protocol)
,其中 transport 支持 ReadTransport
接口而 protocol 是由 protocol_factory 所实例化的对象。
使用 SelectorEventLoop
事件循环, pipe 被设置为非阻塞模式。
coroutine loop.connect_write_pipe(protocol_factory, pipe)
在事件循环中注册 pipe 的写入端。
protocol_factory 必须为一个返回 asyncio 协议 实现的可调用对象。
pipe 是个 类似文件型对象.
返回一对 (transport, protocol)
,其中 transport 支持 WriteTransport
接口而 protocol 是由 protocol_factory 所实例化的对象。
使用 SelectorEventLoop
事件循环, pipe 被设置为非阻塞模式。
在 Windows 中 SelectorEventLoop
不支持上述方法。 对于 Windows 请改用 ProactorEventLoop
。
loop.subprocess_exec()
和 loop.subprocess_shell()
方法。
loop.add_signal_handler(signum, callback, *args)
设置 callback 作为 signum 信号的处理程序。
此回调将与该事件循环中其他加入队列的回调和可运行协程一起由 loop 发起调用。 不同与使用 signal.signal()
注册的信号处理程序,使用此函数注册的回调可以与事件循环进行交互。
如果信号数字非法或者不可捕获,就抛出一个 ValueError
。如果建立处理器的过程中出现问题,会抛出一个 RuntimeError
。
使用 functools.partial()
传递关键字参数 给 callback.
和 signal.signal()
一样,这个函数只能在主线程中调用。
loop.remove_signal_handler(sig)
移除 sig 信号的处理程序。
如果信号处理程序被移除则返回 True
,否则如果给定信号未设置处理程序则返回 False
。
可用性: Unix。
awaitable loop.run_in_executor(executor, func, *args)
安排在指定的执行器中调用 func 。
The executor argument should be an concurrent.futures.Executor
instance. The default executor is used if executor is None
.
import asyncio
import concurrent.futures
def blocking_io():
# File operations (such as logging) can block the
# event loop: run them in a thread pool.
with open('/dev/urandom', 'rb') as f:
return f.read(100)
def cpu_bound():
# CPU-bound operations will block the event loop:
# in general it is preferable to run them in a
# process pool.
return sum(i * i for i in range(10 ** 7))
async def main():
loop = asyncio.get_running_loop()
## Options:
# 1. Run in the default loop's executor:
result = await loop.run_in_executor(
None, blocking_io)
print('default thread pool', result)
# 2. Run in a custom thread pool:
with concurrent.futures.ThreadPoolExecutor() as pool:
result = await loop.run_in_executor(
pool, blocking_io)
print('custom thread pool', result)
# 3. Run in a custom process pool:
with concurrent.futures.ProcessPoolExecutor() as pool:
result = await loop.run_in_executor(
pool, cpu_bound)
print('custom process pool', result)
if __name__ == '__main__':
asyncio.run(main())
Note that the entry point guard (if __name__ == '__main__'
)
is required for option 3 due to the peculiarities of multiprocessing
,
which is used by ProcessPoolExecutor
.
See Safe importing of main module.
这个方法返回一个 asyncio.Future
对象。
使用 functools.partial()
传递关键字参数 给 func 。
在 3.5.3 版更改: loop.run_in_executor()
不会再配置它所创建的线程池执行器的 max_workers
,而是将其留给线程池执行器 (ThreadPoolExecutor
) 来设置默认值。
loop.set_default_executor(executor)
Set executor as the default executor used by run_in_executor()
.
executor must be an instance of
ThreadPoolExecutor
.
在 3.11 版更改: executor must be an instance of
ThreadPoolExecutor
.
loop.default_exception_handler(context)
默认的异常处理器。
此方法会在发生异常且未设置异常处理程序时被调用。 此方法也可以由想要具有不同于默认处理程序的行为的自定义异常处理程序来调用。
context 参数和 call_exception_handler()
中的同名参数完全相同。
loop.call_exception_handler(context)
调用当前事件循环的异常处理器。
context 是个包含下列键的 dict
对象(未来版本的Python可能会引入新键):
'message': 错误消息;
'exception' (可选): 异常对象;
'future' (可选): asyncio.Future
实例;
'task' (可选): asyncio.Task
实例;
'handle' (可选): asyncio.Handle
实例;
'protocol' (可选): Protocol 实例;
'transport' (可选): Transport 实例;
'socket' (可选): socket.socket
实例;
'asyncgen' (可选): 异步生成器,它导致了这个异常
本小节所描述的方法都是低层级的。 在常规 async/await 代码中请考虑改用高层级的 asyncio.create_subprocess_shell()
和 asyncio.create_subprocess_exec()
便捷函数。
在 Windows 中,默认的事件循环 ProactorEventLoop
支持子进程,而 SelectorEventLoop
不支持。参见 Windows 中的子进程支持 。
coroutine loop.subprocess_exec(protocol_factory, *args, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, **kwargs)
用 args 指定的一个或者多个字符串型参数创建一个子进程。
args 必须是个由下列形式的字符串组成的列表:
str
;
或者由 文件系统编码格式 编码的 bytes
。
第一个字符串指定可执行程序,其余的字符串指定其参数。 所有字符串参数共同组成了程序的 argv
。
此方法类似于调用标准库 subprocess.Popen
类,设置 shell=False
并将字符串列表作为第一个参数传入;但是,Popen
只接受一个单独的字符串列表参数,而 subprocess_exec 接受多个字符串参数。
protocol_factory 必须为一个返回 asyncio.SubprocessProtocol
类的子类的可调用对象。
其他参数:
stdin 可以是以下对象之一:
一个文件类对象,表示要使用 connect_write_pipe()
连接到子进程的标准输入流的管道
subprocess.PIPE
常量(默认),将创建并连接一个新的管道。
None
值,这将使得子进程继承来自此进程的文件描述符
subprocess.DEVNULL
常量,这表示将使用特殊的 os.devnull
文件
stdout 可以是以下对象之一:
一个文件类对象,表示要使用 connect_write_pipe()
连接到子进程的标准输出流的管道
subprocess.PIPE
常量(默认),将创建并连接一个新的管道。
None
值,这将使得子进程继承来自此进程的文件描述符
subprocess.DEVNULL
常量,这表示将使用特殊的 os.devnull
文件
stderr 可以是以下对象之一:
一个文件类对象,表示要使用 connect_write_pipe()
连接到子进程的标准错误流的管道
subprocess.PIPE
常量(默认),将创建并连接一个新的管道。
None
值,这将使得子进程继承来自此进程的文件描述符
subprocess.DEVNULL
常量,这表示将使用特殊的 os.devnull
文件
subprocess.STDOUT
常量,将把标准错误流连接到进程的标准输出流
所有其他关键字参数会被不加解释地传给 subprocess.Popen
,除了 bufsize, universal_newlines, shell, text, encoding 和 errors,它们都不应当被指定。
asyncio
子进程 API 不支持将流解码为文本。 可以使用 bytes.decode()
来将从流返回的字节串转换为文本。
其他参数的文档,请参阅 subprocess.Popen
类的构造函数。
返回一对 (transport, protocol)
,其中 transport 来自 asyncio.SubprocessTransport
基类而 protocol 是由 protocol_factory 所实例化的对象。
coroutine loop.subprocess_shell(protocol_factory, cmd, *, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, **kwargs)
基于 cmd 创建一个子进程,该参数可以是一个 str
或者按 文件系统编码格式 编码得到的 bytes
,使用平台的 "shell" 语法。
这类似与用 shell=True
调用标准库的 subprocess.Popen
类。
protocol_factory 必须为一个返回 SubprocessProtocol
类的子类的可调用对象。
请参阅 subprocess_exec()
了解有关其余参数的详情。
返回一对 (transport, protocol)
,其中 transport 来自 SubprocessTransport
基类而 protocol 是由 protocol_factory 所实例化的对象。
应用程序要负责确保正确地转义所有空白字符和特殊字符以防止 shell 注入 漏洞。 shlex.quote()
函数可以被用来正确地转义字符串中可能被用来构造 shell 命令的空白字符和特殊字符。
class asyncio.TimerHandle
由 loop.call_later()
和 loop.call_at()
所返回的回调包装器对象。
这个类是 Handle
的子类。
when()
返回加入计划任务的回调时间,以 float
值表示的秒数。
时间值是一个绝对时间戳,使用与 loop.time()
相同的时间引用。
3.7 新版功能.
Server 对象
Server 对象可使用 loop.create_server()
, loop.create_unix_server()
, start_server()
和 start_unix_server()
等函数来创建。
Do not instantiate the Server
class directly.
class asyncio.Server
Server 对象是异步上下文管理器。当用于 async with
语句时,异步上下文管理器可以确保 Server 对象被关闭,并且在 async with
语句完成后,不接受新的连接。
srv = await loop.create_server(...
)
async with srv:
# some code
# At this point, srv is closed and no longer accepts new connections.
在 3.7 版更改: Python3.7 开始,Server 对象是一个异步上下文管理器。
在 3.11 版更改: This class was exposed publicly as asyncio.Server
in Python 3.9.11, 3.10.3 and 3.11.
close()
停止服务:关闭监听的套接字并且设置 sockets
属性为 None
。
用于表示已经连进来的客户端连接会保持打开的状态。
服务器是被异步关闭的,使用 wait_closed()
协程来等待服务器关闭。
This method is idempotent, so it can be called when
the server is already serving.
传给 loop.create_server()
和 asyncio.start_server()
的 start_serving 仅限关键字形参允许创建不接受初始连接的 Server 对象。 在此情况下可以使用 Server.start_serving()
或 Server.serve_forever()
让 Server 对象开始接受连接。
3.7 新版功能.
coroutine serve_forever()
开始接受连接,直到协程被取消。 serve_forever
任务的取消将导致服务器被关闭。
如果服务器已经在接受连接了,这个方法可以被调用。每个 Server 对象,仅能有一个 serve_forever
任务。
async def client_connected(reader, writer):
# Communicate with the client with
# reader/writer streams. For example:
await reader.readline()
async def main(host, port):
srv = await asyncio.start_server(
client_connected, host, port)
await srv.serve_forever()
asyncio.run(main('127.0.0.1', 0))
3.7 新版功能.
sockets
List of socket-like objects, asyncio.trsock.TransportSocket
, which
the server is listening on.
在 3.7 版更改: 在 Python 3.7 之前 Server.sockets
会直接返回内部的服务器套接字列表。 在 3.7 版则会返回该列表的副本。
事件循环实现
asyncio 带有两种不同的事件循环实现: SelectorEventLoop
和 ProactorEventLoop
。
默认情况下 asyncio 被配置为在 Unix 上使用 SelectorEventLoop
而在 Windows 上使用 ProactorEventLoop
。
class asyncio.SelectorEventLoop
基于 selectors
模块的事件循环。
使用给定平台中最高效的可用 selector。 也可以手动配置要使用的特定 selector:
import asyncio
import selectors
class MyPolicy(asyncio.DefaultEventLoopPolicy):
def new_event_loop(self):
selector = selectors.SelectSelector()
return asyncio.SelectorEventLoop(selector)
asyncio.set_event_loop_policy(MyPolicy())
可用性: Unix, Windows。
asyncio 兼容事件循环的抽象基类。
The 事件循环方法集 section lists all
methods that an alternative implementation of AbstractEventLoop
should have defined.
请注意本节中的所有示例都 有意地 演示了如何使用低层级的事件循环 API,例如 loop.run_forever()
和 loop.call_soon()
。 现代的 asyncio 应用很少需要以这样的方式编写;请考虑使用高层级的函数例如 asyncio.run()
。
call_soon() 的 Hello World 示例。
一个使用 loop.call_soon()
方法来安排回调的示例。 回调会显示 "Hello World"
然后停止事件循环:
import asyncio
def hello_world(loop):
"""A callback to print 'Hello World' and stop the event loop"""
print('Hello World')
loop.stop()
loop = asyncio.new_event_loop()
# Schedule a call to hello_world()
loop.call_soon(hello_world, loop)
# Blocking call interrupted by loop.stop()
try:
loop.run_forever()
finally:
loop.close()
一个类似的 Hello World 示例,使用协程和 run()
函数创建。
使用 call_later() 来展示当前的日期
一个每秒刷新显示当前日期的示例。 回调使用 loop.call_later()
方法在 5 秒后将自身重新加入计划日程,然后停止事件循环:
import asyncio
import datetime
def display_date(end_time, loop):
print(datetime.datetime.now())
if (loop.time() + 1.0) < end_time:
loop.call_later(1, display_date, end_time, loop)
else:
loop.stop()
loop = asyncio.new_event_loop()
# Schedule the first call to display_date()
end_time = loop.time() + 5.0
loop.call_soon(display_date, end_time, loop)
# Blocking call interrupted by loop.stop()
try:
loop.run_forever()
finally:
loop.close()
一个类似的 current date 示例,使用协程和 run()
函数创建。
监控一个文件描述符的读事件
使用 loop.add_reader()
方法,等到文件描述符收到一些数据,然后关闭事件循环:
import asyncio
from socket import socketpair
# Create a pair of connected file descriptors
rsock, wsock = socketpair()
loop = asyncio.new_event_loop()
def reader():
data = rsock.recv(100)
print("Received:", data.decode())
# We are done: unregister the file descriptor
loop.remove_reader(rsock)
# Stop the event loop
loop.stop()
# Register the file descriptor for read event
loop.add_reader(rsock, reader)
# Simulate the reception of data from the network
loop.call_soon(wsock.send, 'abc'.encode())
try:
# Run the event loop
loop.run_forever()
finally:
# We are done. Close sockets and the event loop.
rsock.close()
wsock.close()
loop.close()
一个类似的 示例,使用传输、协议和 loop.create_connection()
方法创建。
另一个类似的 示例,使用了高层级的 asyncio.open_connection()
函数和流。
为SIGINT和SIGTERM设置信号处理器
(这个 signals
示例只适用于 Unix。)
使用 loop.add_signal_handler()
方法为信号 SIGINT
和 SIGTERM
注册处理程序:
import asyncio
import functools
import os
import signal
def ask_exit(signame, loop):
print("got signal %s: exit" % signame)
loop.stop()
async def main():
loop = asyncio.get_running_loop()
for signame in {'SIGINT', 'SIGTERM'}:
loop.add_signal_handler(
getattr(signal, signame),
functools.partial(ask_exit, signame, loop))
await asyncio.sleep(3600)
print("Event loop running for 1 hour, press Ctrl+C to interrupt.")
print(f"pid {os.getpid()}: send SIGINT or SIGTERM to exit.")
asyncio.run(main())
This page is licensed under the Python Software Foundation License Version 2.
Examples, recipes, and other code in the documentation are additionally licensed under the Zero Clause BSD License.
See History and License for more information.
The Python Software Foundation is a non-profit corporation.
Please donate.
最后更新于 9月 12, 2023.
Found a bug?