Цикл подій¶
Вихідний код: Lib/asyncio/events.py, Lib/asyncio/base_events.py
Передмова
Цикл подій є ядром кожної асинхронної програми. Цикли подій запускають асинхронні завдання та зворотні виклики, виконують мережеві операції вводу-виводу та запускають підпроцеси.
Розробникам додатків зазвичай слід використовувати асинхронні функції високого рівня, такі як asyncio.run()
, і їм рідко потрібно посилатися на об’єкт циклу або викликати його методи. Цей розділ призначений переважно для авторів коду нижчого рівня, бібліотек і фреймворків, яким потрібен точніший контроль над поведінкою циклу подій.
Отримання циклу подій
Наступні функції низького рівня можна використовувати для отримання, встановлення або створення циклу подій:
-
asyncio.
get_running_loop
()¶ Повернути запущений цикл подій у поточному потоці ОС.
If there is no running event loop a
RuntimeError
is raised. This function can only be called from a coroutine or a callback.Нове в версії 3.7.
-
asyncio.
get_event_loop
()¶ Отримати поточний цикл подій.
If there is no current event loop set in the current OS thread, the OS thread is main, and
set_event_loop()
has not yet been called, asyncio will create a new event loop and set it as the current one.Оскільки ця функція має досить складну поведінку (особливо, коли використовуються користувацькі політики циклу подій), використанню функції
get_running_loop()
краще, ніжget_event_loop()
у співпрограмах і зворотних викликах.Consider also using the
asyncio.run()
function instead of using lower level functions to manually create and close an event loop.
-
asyncio.
set_event_loop
(loop)¶ Set loop as a current event loop for the current OS thread.
-
asyncio.
new_event_loop
()¶ Створити та повернути новий об’єкт циклу подій.
Зауважте, що поведінку функцій get_event_loop()
, set_event_loop()
і new_event_loop()
можна змінити шляхом встановлення спеціальної політики циклу подій.
Зміст
Ця сторінка документації містить такі розділи:
Розділ Event Loop Methods є довідковою документацією щодо API циклу подій;
Розділ Callback Handles документує екземпляри
Handle
іTimerHandle
, які повертаються з таких методів планування, якloop.call_soon()
іloop.call_later()
;Розділ Server Objects документує типи, що повертаються з методів циклу подій, таких як
loop.create_server()
;Розділ Event Loop Implementations документує класи
SelectorEventLoop
іProactorEventLoop
;Розділ Examples демонструє, як працювати з деякими API циклу подій.
Методи циклу подій¶
Цикли подій мають API низького рівня для наступного:
Запуск і зупинка циклу¶
-
loop.
run_until_complete
(future)¶ Виконуйте, доки future (примірник
Future
) не завершиться.Якщо аргумент є об’єктом співпрограми, він неявно запланований для виконання як
asyncio.Task
.Повернути результат майбутнього або викликати його виключення.
-
loop.
run_forever
()¶ Запускайте цикл подій, доки не буде викликано
stop()
.If
stop()
is called beforerun_forever()
is called, the loop will poll the I/O selector once with a timeout of zero, run all callbacks scheduled in response to I/O events (and those that were already scheduled), and then exit.Якщо
stop()
викликається під час роботиrun_forever()
, цикл запустить поточний пакет зворотних викликів, а потім завершить роботу. Зауважте, що нові зворотні виклики, заплановані зворотними викликами, не виконуватимуться в цьому випадку; натомість вони запустяться під час наступного викликуrun_forever()
абоrun_until_complete()
.
-
loop.
stop
()¶ Зупиніть цикл подій.
-
loop.
is_running
()¶ Повертає
True
, якщо цикл подій зараз запущено.
-
loop.
is_closed
()¶ Повертає
True
, якщо цикл події було закрито.
-
loop.
close
()¶ Закрийте цикл подій.
Під час виклику цієї функції цикл не повинен працювати. Усі зворотні виклики, що очікують на розгляд, буде відхилено.
Цей метод очищає всі черги та вимикає виконавець, але не чекає, поки виконавець завершить роботу.
Цей метод є ідемпотентним і необоротним. Ніякі інші методи не повинні викликатися після закриття циклу подій.
-
coroutine
loop.
shutdown_asyncgens
()¶ Schedule all currently open asynchronous generator objects to close with an
aclose()
call. After calling this method, the event loop will issue a warning if a new asynchronous generator is iterated. This should be used to reliably finalize all scheduled asynchronous generators.Зауважте, що немає потреби викликати цю функцію, коли використовується
asyncio.run()
.Приклад:
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
. After calling this method, aRuntimeError
will be raised ifloop.run_in_executor()
is called while using the default executor.Зауважте, що немає потреби викликати цю функцію, коли використовується
asyncio.run()
.Нове в версії 3.9.
Планування зворотних дзвінків¶
-
loop.
call_soon
(callback, *args, context=None)¶ Заплануйте виклик callback callback з аргументами args на наступній ітерації циклу подій.
Зворотні виклики викликаються в тому порядку, в якому вони зареєстровані. Кожен зворотній виклик буде викликано рівно один раз.
Необов’язковий аргумент context, що містить лише ключове слово, дозволяє вказати спеціальний
contextvars.Context
для виконання зворотного виклику. Поточний контекст використовується, якщо контексту не надано.An instance of
asyncio.Handle
is returned, which can be used later to cancel the callback.This method is not thread-safe.
-
loop.
call_soon_threadsafe
(callback, *args, context=None)¶ A thread-safe variant of
call_soon()
. Must be used to schedule callbacks from another thread.Викликає
RuntimeError
, якщо викликається в закритому циклі. Це може статися у вторинному потоці, коли основна програма вимикається.Перегляньте розділ паралелізм і багатопотоковість документації.
Змінено в версії 3.7: Додано параметр context тільки для ключового слова. Дивіться PEP 567 для більш детальної інформації.
Примітка
Більшість функцій планування asyncio
не дозволяють передавати ключові аргументи. Для цього скористайтеся functools.partial()
:
# will schedule "print("Hello", flush=True)"
loop.call_soon(
functools.partial(print, "Hello", flush=True))
Використання часткових об’єктів зазвичай зручніше, ніж використання лямбда-виразів, оскільки asyncio може краще відтворювати часткові об’єкти в повідомленнях про налагодження та помилки.
Планування відкладених зворотних викликів¶
Цикл подій надає механізми для планування виклику функцій зворотного виклику в певний момент у майбутньому. Цикл подій використовує монотонні годинники для відстеження часу.
-
loop.
call_later
(delay, callback, *args, context=None)¶ Розклад зворотного виклику для виклику після заданої затримки у секундах (може бути як int, так і float).
Повертається екземпляр
asyncio.TimerHandle
, який можна використовувати для скасування зворотного виклику.callback буде передзвонено рівно один раз. Якщо два зворотні виклики заплановано на один і той же час, порядок їх викликів не визначений.
Додатковий позиційний args буде передано зворотному виклику під час його виклику. Якщо ви хочете, щоб зворотній виклик викликався з аргументами ключового слова, використовуйте
functools.partial()
.Необов’язковий аргумент context, що містить лише ключове слово, дозволяє вказати спеціальний
contextvars.Context
для виконання зворотного виклику. Поточний контекст використовується, якщо контексту не надано.Змінено в версії 3.7: Додано параметр context тільки для ключового слова. Дивіться PEP 567 для більш детальної інформації.
Змінено в версії 3.8: У Python 3.7 і раніше з реалізацією циклу подій за замовчуванням затримка не могла перевищувати один день. Це було виправлено в 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.
time
()¶ Повертає поточний час у вигляді значення
float
відповідно до внутрішнього монотонного годинника циклу подій.
Примітка
Змінено в версії 3.8: У Python 3.7 та попередніх версіях тайм-аути (відносна затримка або абсолютна коли) не повинні перевищувати одного дня. Це було виправлено в Python 3.8.
Дивись також
Функція asyncio.sleep()
.
Створення ф’ючерсів і завдань¶
-
loop.
create_future
()¶ Створіть об’єкт
asyncio.Future
, приєднаний до циклу подій.Це найкращий спосіб створення ф’ючерсів в асинхронному режимі. Це дозволяє стороннім циклам подій надавати альтернативні реалізації об’єкта Future (з кращою продуктивністю або інструментарієм).
Нове в версії 3.5.2.
-
loop.
create_task
(coro, *, name=None)¶ Schedule the execution of a Співпрограми. Return a
Task
object.Сторонні цикли подій можуть використовувати власний підклас
Task
для взаємодії. У цьому випадку тип результату є підкласомTask
.Якщо вказано аргумент name, а не
None
, він встановлюється як назва завдання за допомогоюTask.set_name()
.Змінено в версії 3.8: Added the
name
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)
, where loop is a reference to the active event loop, and coro is a coroutine object. The callable must return aasyncio.Future
-compatible object.
-
loop.
get_task_factory
()¶ Повертає фабрику завдань або
None
, якщо використовується типова.
Відкриття мережевих підключень¶
-
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, happy_eyeballs_delay=None, interleave=None)¶ Відкрийте потокове транспортне з’єднання за вказаною адресою, указаною host і port.
The socket family can be either
AF_INET
orAF_INET6
depending on host (or the family argument, if provided).The socket type will be
SOCK_STREAM
.protocol_factory має бути викликом, що повертає реалізацію asyncio protocol.
Цей метод намагатиметься встановити з’єднання у фоновому режимі. У разі успіху він повертає пару
(транспорт, протокол)
.Хронологічний синопсис основної операції такий:
З’єднання встановлюється та для нього створюється транспорт.
protocol_factory викликається без аргументів і має повернути екземпляр protocol.
Екземпляр протоколу з’єднується з транспортом шляхом виклику його методу
connection_made()
.Кортеж
(транспорт, протокол)
повертається в разі успіху.
Створений транспорт є двонаправленим потоком, що залежить від реалізації.
Інші аргументи:
ssl: якщо задано і не має значення false, створюється транспорт SSL/TLS (за замовчуванням створюється звичайний транспорт TCP). Якщо ssl є об’єктом
ssl.SSLContext
, цей контекст використовується для створення транспорту; якщо ssl має значенняTrue
, використовується контекст за замовчуванням, який повертається зssl.create_default_context()
.Дивись також
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. Типовим значенням є0
, якщо happy_eyeballs_delay не вказано, і1
, якщо так.sock, якщо його надано, має бути існуючим, уже підключеним
socket.socket
об’єктом, який буде використовуватися транспортом. Якщо вказано sock, жоден з host, port, family, proto, flags, happy_eyeballs_delay, interleave і local_addr не повинен бути вказаний.local_addr, якщо вказано, це кортеж
(local_host, local_port)
, який використовується для локального зв’язування сокета. local_host і local_port шукаються за допомогоюgetaddrinfo()
, подібно до host і port.ssl_handshake_timeout — це (для з’єднання TLS) час у секундах очікування завершення рукостискання TLS перед перериванням з’єднання.
60.0
секунд, якщоNone
(за замовчуванням).
Нове в версії 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://tools.ietf.org/html/rfc6555
Нове в версії 3.7: The ssl_handshake_timeout parameter.
Змінено в версії 3.6: The socket option
TCP_NODELAY
is set by default for all TCP connections.Змінено в версії 3.5: Додано підтримку SSL/TLS у
ProactorEventLoop
.Дивись також
Функція
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_address=None, reuse_port=None, allow_broadcast=None, sock=None)¶ Примітка
The parameter reuse_address is no longer supported, as using
SO_REUSEADDR
poses a significant security concern for UDP. Explicitly passingreuse_address=True
will raise an exception.Коли кілька процесів з різними UID призначають сокети ідентичній адресі сокета UDP за допомогою
SO_REUSEADDR
, вхідні пакети можуть розподілятися між сокетами випадковим чином.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.Створіть з’єднання дейтаграми.
The socket family can be either
AF_INET
,AF_INET6
, orAF_UNIX
, depending on host (or the family argument, if provided).The socket type will be
SOCK_DGRAM
.protocol_factory має бути викликом, що повертає реалізацію 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
).
Перегляньте приклади UDP echo client protocol і UDP echo server protocol прикладів.
Змінено в версії 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 due to security concerns.
Змінено в версії 3.8: Додана підтримка Windows.
-
coroutine
loop.
create_unix_connection
(protocol_factory, path=None, *, ssl=None, sock=None, server_hostname=None, ssl_handshake_timeout=None)¶ Створіть підключення Unix.
The socket family will be
AF_UNIX
; socket type will beSOCK_STREAM
.Кортеж
(транспорт, протокол)
повертається в разі успіху.path — це ім’я сокета домену Unix і є обов’язковим, якщо не вказано параметр sock. Підтримуються абстрактні сокети Unix, шляхи
str
,bytes
іPath
.Перегляньте документацію методу
loop.create_connection()
, щоб отримати інформацію про аргументи цього методу.Наявність: Unix.
Нове в версії 3.7: The ssl_handshake_timeout parameter.
Змінено в версії 3.7: The path parameter can now be a path-like object.
Створення мережевих серверів¶
-
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, start_serving=True)¶ Create a TCP server (socket type
SOCK_STREAM
) listening on port of the host address.Повертає об’єкт
Server
.Аргументи:
protocol_factory має бути викликом, що повертає реалізацію protocol.
Для параметра host можна встановити кілька типів, які визначають, де сервер буде слухати:
Якщо host є рядком, сервер TCP прив’язаний до єдиного мережевого інтерфейсу, визначеного host.
Якщо host є послідовністю рядків, TCP-сервер прив’язаний до всіх мережевих інтерфейсів, визначених цією послідовністю.
Якщо host є порожнім рядком або
None
, усі інтерфейси передбачаються, і буде повернено список кількох сокетів (швидше за все, один для IPv4 та інший для IPv6).
Параметр port можна встановити, щоб вказати, який порт сервер повинен слухати. Якщо
0
абоNone
(за замовчуванням), буде вибрано випадковий невикористаний порт (зауважте, що якщо host розпізнає кілька мережевих інтерфейсів, для кожного інтерфейсу буде вибрано окремий випадковий порт).family can be set to either
socket.AF_INET
orAF_INET6
to force the socket to use IPv4 or IPv6. If not set, the family will be determined from host name (defaults toAF_UNSPEC
).flags — це бітова маска для
getaddrinfo()
.За бажанням можна вказати sock, щоб використовувати вже існуючий об’єкт socket. Якщо вказано, host і port не повинні вказуватися.
backlog — це максимальна кількість підключень у черзі, переданих до
listen()
(за замовчуванням 100).ssl можна встановити як екземпляр
SSLContext
, щоб увімкнути TLS через прийнятні з’єднання.reuse_address повідомляє ядру повторно використовувати локальний сокет у стані
TIME_WAIT
, не чекаючи закінчення його природного часу очікування. Якщо не вказано, для Unix буде автоматично встановлено значенняTrue
.reuse_port повідомляє ядру дозволити цю кінцеву точку прив’язувати до того самого порту, до якого прив’язані інші існуючі кінцеві точки, за умови, що всі вони встановлюють цей прапор під час створення. Цей параметр не підтримується в Windows.
ssl_handshake_timeout — це (для TLS-сервера) час у секундах очікування завершення рукостискання TLS перед розривом з’єднання.
60.0
секунд, якщоNone
(за замовчуванням).start_serving встановлений на
True
(за замовчуванням), змушує створений сервер негайно приймати підключення. Якщо встановлено значенняFalse
, користувач повинен чекатиServer.start_serving()
абоServer.serve_forever()
, щоб змусити сервер почати приймати з’єднання.
Нове в версії 3.7: Added ssl_handshake_timeout and start_serving parameters.
Змінено в версії 3.6: The socket option
TCP_NODELAY
is set by default for all TCP connections.Змінено в версії 3.5: Додано підтримку SSL/TLS у
ProactorEventLoop
.Змінено в версії 3.5.1: Параметр host може бути послідовністю рядків.
Дивись також
Функція
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, start_serving=True)¶ Similar to
loop.create_server()
but works with theAF_UNIX
socket family.path — це ім’я сокета домену Unix і є обов’язковим, якщо не надано аргумент sock. Підтримуються абстрактні сокети Unix, шляхи
str
,bytes
іPath
.Перегляньте документацію методу
loop.create_server()
для отримання інформації про аргументи цього методу.Наявність: Unix.
Нове в версії 3.7: The ssl_handshake_timeout and start_serving parameters.
Змінено в версії 3.7: The path parameter can now be a
Path
object.
-
coroutine
loop.
connect_accepted_socket
(protocol_factory, sock, *, ssl=None, ssl_handshake_timeout=None)¶ Оберніть уже прийняте підключення до пари транспорт/протокол.
Цей метод може використовуватися серверами, які приймають з’єднання за межами asyncio, але використовують asyncio для їх обробки.
Параметри:
protocol_factory має бути викликом, що повертає реалізацію protocol.
sock — це вже існуючий об’єкт сокета, який повертається з
socket.accept
.ssl можна встановити як
SSLContext
, щоб увімкнути SSL через прийнятні з’єднання.ssl_handshake_timeout — це (для з’єднання SSL) час у секундах очікування завершення рукостискання SSL перед розривом з’єднання.
60.0
секунд, якщоNone
(за замовчуванням).
Повертає пару
(транспорт, протокол)
.Нове в версії 3.7: The ssl_handshake_timeout parameter.
Нове в версії 3.5.3.
Передача файлів¶
-
coroutine
loop.
sendfile
(transport, file, offset=0, count=None, *, fallback=True)¶ Надіслати файл через транспорт. Повертає загальну кількість надісланих байтів.
Метод використовує високопродуктивний
os.sendfile()
, якщо він доступний.file має бути звичайним файловим об’єктом, відкритим у двійковому режимі.
offset вказує, звідки почати читання файлу. Якщо вказано, count — це загальна кількість байтів для передачі, а не надсилання файлу до досягнення EOF. Позиція файлу завжди оновлюється, навіть якщо цей метод викликає помилку, і
file.tell()
можна використовувати для отримання фактичної кількості надісланих байтів.fallback встановлений на
True
робить asyncio для ручного читання та надсилання файлу, коли платформа не підтримує системний виклик sendfile (наприклад, Windows або SSL-сокет на Unix).Викликати
SendfileNotAvailableError
, якщо система не підтримує системний виклик sendfile і fallback має значенняFalse
.Нове в версії 3.7.
Оновлення TLS¶
-
coroutine
loop.
start_tls
(transport, protocol, sslcontext, *, server_side=False, server_hostname=None, ssl_handshake_timeout=None)¶ Оновіть існуюче транспортне підключення до TLS.
Return a new transport instance, that the protocol must start using immediately after the await. The transport instance passed to the start_tls method should never be used again.
Параметри:
екземпляри transport і protocol, які повертають такі методи, як
create_server()
іcreate_connection()
.sslcontext: налаштований екземпляр
SSLContext
.server_side передає
True
, коли з’єднання на стороні сервера оновлюється (наприклад, створенеcreate_server()
).server_hostname: встановлює або замінює ім’я хоста, з яким буде зіставлятися сертифікат цільового сервера.
ssl_handshake_timeout — це (для з’єднання TLS) час у секундах очікування завершення рукостискання TLS перед перериванням з’єднання.
60.0
секунд, якщоNone
(за замовчуванням).
Нове в версії 3.7.
Перегляд дескрипторів файлів¶
-
loop.
add_reader
(fd, callback, *args)¶ Розпочніть моніторинг дескриптора файлу fd на доступність читання та викличте callback із зазначеними аргументами, коли fd стане доступним для читання.
-
loop.
remove_reader
(fd)¶ Stop monitoring the fd file descriptor for read availability.
-
loop.
add_writer
(fd, callback, *args)¶ Розпочніть моніторинг дескриптора файлу fd на доступність запису та викличте callback із зазначеними аргументами, коли fd стане доступним для запису.
Використовуйте
functools.partial()
, щоб передати аргументи ключового слова в callback.
-
loop.
remove_writer
(fd)¶ Stop monitoring the fd file descriptor for write availability.
Перегляньте також розділ Підтримка платформи, щоб дізнатися про деякі обмеження цих методів.
Безпосередня робота з об’єктами сокетів¶
Загалом, реалізації протоколів, які використовують API на основі транспорту, такі як loop.create_connection()
і loop.create_server()
, є швидшими, ніж реалізації, які працюють безпосередньо з сокетами. Однак є деякі випадки використання, коли продуктивність не є критичною, і працювати з об’єктами socket
напряму зручніше.
-
coroutine
loop.
sock_recv
(sock, nbytes)¶ Отримайте до nbytes від sock. Асинхронна версія
socket.recv()
.Повернути отримані дані як об’єкт bytes.
sock має бути неблокуючим сокетом.
Змінено в версії 3.7: Незважаючи на те, що цей метод завжди документувався як метод співпрограми, випуски до Python 3.7 повертали
Future
. Починаючи з Python 3.7, це методasync def
.
-
coroutine
loop.
sock_recv_into
(sock, buf)¶ Отримувати дані з sock в буфер buf. Створено за методом блокування
socket.recv_into()
.Повертає кількість байтів, записаних у буфер.
sock має бути неблокуючим сокетом.
Нове в версії 3.7.
-
coroutine
loop.
sock_sendall
(sock, data)¶ Надішліть дані в сокет sock. Асинхронна версія
socket.sendall()
.Цей метод продовжує надсилати дані в сокет, доки не буде надіслано всі дані в data або не станеться помилка.
None
повертається в разі успіху. У разі помилки виникає виняток. Крім того, немає способу визначити, скільки даних, якщо такі були, було успішно оброблено приймальною стороною з’єднання.sock має бути неблокуючим сокетом.
Змінено в версії 3.7: Even though the method was always documented as a coroutine method, before Python 3.7 it returned an
Future
. Since Python 3.7, this is anasync def
method.
-
coroutine
loop.
sock_connect
(sock, address)¶ Підключіть sock до віддаленої розетки за адресою.
Асинхронна версія
socket.connect()
.sock має бути неблокуючим сокетом.
Змінено в версії 3.5.2:
address
більше не потребує вирішення.sock_connect
спробує перевірити, чи адреса вже дозволена, викликавшиsocket.inet_pton()
. Якщо ні,loop.getaddrinfo()
буде використано для визначення адреси.Дивись також
-
coroutine
loop.
sock_accept
(sock)¶ Прийняти підключення. Створено за методом блокування
socket.accept()
.Сокет має бути прив’язаний до адреси та прослуховувати підключення. Поверненим значенням є пара
(conn, address)
, де conn — це новий об’єкт сокета, який можна використовувати для надсилання та отримання даних під час з’єднання, а address — це адреса, прив’язана до сокета на іншому кінець з’єднання.sock має бути неблокуючим сокетом.
Змінено в версії 3.7: Незважаючи на те, що метод завжди документувався як метод співпрограми, до Python 3.7 він повертав
Future
. Починаючи з Python 3.7, це методasync def
.Дивись також
-
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 або сокет SSL в Unix).Викликати
SendfileNotAvailableError
, якщо система не підтримує системний виклик sendfile і fallback має значенняFalse
.sock має бути неблокуючим сокетом.
Нове в версії 3.7.
DNS¶
-
coroutine
loop.
getaddrinfo
(host, port, *, family=0, type=0, proto=0, flags=0)¶ Асинхронна версія
socket.getaddrinfo()
.
-
coroutine
loop.
getnameinfo
(sockaddr, flags=0)¶ Асинхронна версія
socket.getnameinfo()
.
Змінено в версії 3.7: Обидва методи getaddrinfo і getnameinfo завжди були задокументовані для повернення співпрограми, але до Python 3.7 вони фактично повертали об’єкти asyncio.Future
. Починаючи з Python 3.7 обидва методи є співпрограмами.
Робота з трубами¶
-
coroutine
loop.
connect_read_pipe
(protocol_factory, pipe)¶ Зареєструйте прочитаний кінець pipe у циклі подій.
protocol_factory має бути викликом, що повертає реалізацію asyncio protocol.
pipe — це файлоподібний об’єкт.
Повернена пара
(транспорт, протокол)
, де transport підтримує інтерфейсReadTransport
, а protocol є об’єктом, створеним protocol_factory.За допомогою циклу подій
SelectorEventLoop
pipe встановлюється в неблокуючий режим.
-
coroutine
loop.
connect_write_pipe
(protocol_factory, pipe)¶ Зареєструйте кінець запису pipe у циклі подій.
protocol_factory має бути викликом, що повертає реалізацію asyncio protocol.
pipe — це файлоподібний об’єкт.
Повернена пара
(транспорт, протокол)
, де transport підтримує інтерфейсWriteTransport
, а protocol є об’єктом, створеним protocol_factory.За допомогою циклу подій
SelectorEventLoop
pipe встановлюється в неблокуючий режим.
Примітка
SelectorEventLoop
не підтримує наведені вище методи в Windows. Використовуйте ProactorEventLoop
замість цього для Windows.
Дивись також
Сигнали Unix¶
-
loop.
add_signal_handler
(signum, callback, *args)¶ Встановіть callback як обробник для сигналу signum.
Зворотний виклик буде викликаний циклом разом з іншими зворотними викликами в черзі та запущеними співпрограмами цього циклу подій. На відміну від обробників сигналів, зареєстрованих за допомогою
signal.signal()
, зворотній виклик, зареєстрований у цій функції, може взаємодіяти з циклом подій.Викликайте
ValueError
, якщо номер сигналу недійсний або не вловлюється. ВикликатиRuntimeError
, якщо є проблема з налаштуванням обробника.Використовуйте
functools.partial()
, щоб передати аргументи ключового слова в callback.Подібно до
signal.signal()
, ця функція має бути викликана в основному потоці.
-
loop.
remove_signal_handler
(sig)¶ Видаліть обробник для сигналу sig.
Повертає
True
, якщо обробник сигналу було видалено, абоFalse
, якщо обробник не встановлено для даного сигналу.Наявність: Unix.
Дивись також
Модуль signal
.
Виконання коду в потоках або пулах процесів¶
-
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 isNone
.Приклад:
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) asyncio.run(main())
Цей метод повертає об’єкт
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 should be an instance ofThreadPoolExecutor
.Застаріло починаючи з версії 3.8: Using an executor that is not an instance of
ThreadPoolExecutor
is deprecated and will trigger an error in Python 3.9.executor must be an instance of
concurrent.futures.ThreadPoolExecutor
.
API обробки помилок¶
Дозволяє налаштувати спосіб обробки винятків у циклі подій.
-
loop.
set_exception_handler
(handler)¶ Встановіть обробник як новий обробник винятків циклу подій.
Якщо обробник має значення
None
, буде встановлено обробник винятків за умовчанням. В іншому випадку обробник має бути викликом із сигнатурою, що відповідає(цикл, контекст)
, децикл
є посиланням на активний цикл подій, аконтекст
єdict
об’єкт, що містить деталі винятку (перегляньте документаціюcall_exception_handler()
, щоб дізнатися більше про контекст).
-
loop.
get_exception_handler
()¶ Повертає поточний обробник винятків або
None
, якщо настроюваний обробник винятків не встановлено.Нове в версії 3.5.2.
-
loop.
default_exception_handler
(context)¶ Обробник винятків за замовчуванням.
Це викликається, коли виникає виняткова ситуація, а обробник винятків не встановлено. Це може бути викликано спеціальним обробником винятків, який хоче відкласти поведінку обробника за замовчуванням.
Параметр context має те саме значення, що й у
call_exception_handler()
.
-
loop.
call_exception_handler
(context)¶ Викликати обробник винятків поточного циклу подій.
context — це об’єкт
dict
, що містить такі ключі (нові ключі можуть бути представлені в наступних версіях Python):„message“: повідомлення про помилку;
„exception“ (необов’язковий): об’єкт винятку;
„майбутнє“ (необов’язково): екземпляр
asyncio.Future
;„task“ (необов’язковий):
asyncio.Task
екземпляр;„handle“ (необов’язковий):
asyncio.Handle
екземпляр;„протокол“ (необов’язково): примірник протоколу;
„transport“ (необов’язковий): Transport екземпляр;
„socket“ (необов’язковий):
socket.socket
екземпляр;- „asyncgen“ (необов’язково): асинхронний генератор, який викликав
виняток.
Примітка
This method should not be overloaded in subclassed event loops. For custom exception handling, use the
set_exception_handler()
method.
Увімкнення режиму налагодження¶
-
loop.
get_debug
()¶ Отримати режим налагодження (
bool
) циклу подій.Значенням за замовчуванням є
True
, якщо змінна середовищаPYTHONASYNCIODEBUG
має значення непорожнього рядка,False
в іншому випадку.
-
loop.
set_debug
(enabled: bool)¶ Встановіть режим налагодження циклу подій.
Змінено в версії 3.7: Новий Режим розробки Python тепер також можна використовувати для ввімкнення режиму налагодження.
Дивись також
Запущені підпроцеси¶
Методи, описані в цьому підрозділі, є низькорівневими. У звичайному коді 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 може бути будь-яким із цих:
a file-like object representing a pipe to be connected to the subprocess’s standard input stream using
connect_write_pipe()
константа
subprocess.PIPE
(за замовчуванням), яка створить новий канал і з’єднає його,значення
None
, яке змусить підпроцес успадкувати дескриптор файлу від цього процесуконстанта
subprocess.DEVNULL
, яка вказує, що буде використовуватися спеціальний файлos.devnull
stdout може бути будь-яким із цього:
a file-like object representing a pipe to be connected to the subprocess’s standard output stream using
connect_write_pipe()
константа
subprocess.PIPE
(за замовчуванням), яка створить новий канал і з’єднає його,значення
None
, яке змусить підпроцес успадкувати дескриптор файлу від цього процесуконстанта
subprocess.DEVNULL
, яка вказує, що буде використовуватися спеціальний файлos.devnull
stderr може бути будь-яким із цього:
a file-like object representing a pipe to be connected to the subprocess’s standard error stream using
connect_write_pipe()
константа
subprocess.PIPE
(за замовчуванням), яка створить новий канал і з’єднає його,значення
None
, яке змусить підпроцес успадкувати дескриптор файлу від цього процесуконстанта
subprocess.DEVNULL
, яка вказує, що буде використовуватися спеціальний файлos.devnull
константа
subprocess.STDOUT
, яка підключатиме стандартний потік помилок до стандартного потоку виводу процесу
Усі інші аргументи ключових слів передаються до
subprocess.Popen
без інтерпретації, за винятком bufsize, universal_newlines, shell, text, encoding і errors, які не слід вказувати в все.API підпроцесу
asyncio
не підтримує декодування потоків як тексту.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
рядком, закодованим у кодуванні файлової системи, використовуючи синтаксис «оболонки» платформи.Це схоже на клас стандартної бібліотеки
subprocess.Popen
, що викликається за допомогоюshell=True
.protocol_factory має бути викликаним, що повертає підклас класу
SubprocessProtocol
.Перегляньте
subprocess_exec()
для отримання додаткової інформації про інші аргументи.Повертає пару «(transport, protocol)», де transport відповідає базовому класу
SubprocessTransport
, а protocol є об’єктом, створеним protocol_factory.
Примітка
Програма несе відповідальність за те, щоб усі пробіли та спеціальні символи були взяті в лапки належним чином, щоб уникнути вразливості впровадження оболонки. Функцію shlex.quote()
можна використати для правильного екранування пробілів і спеціальних символів у рядках, які використовуватимуться для створення команд оболонки.
Ручки зворотного виклику¶
-
class
asyncio.
Handle
¶ Об’єкт оболонки зворотного виклику, повернутий
loop.call_soon()
,loop.call_soon_threadsafe()
.-
cancel
()¶ Скасувати зворотний дзвінок. Якщо зворотний виклик уже скасовано або виконано, цей метод не має ефекту.
-
cancelled
()¶ Повертає
True
, якщо зворотний виклик було скасовано.Нове в версії 3.7.
-
-
class
asyncio.
TimerHandle
¶ Об’єкт оболонки зворотного виклику, повернутий
loop.call_later()
іloop.call_at()
.Цей клас є підкласом
Handle
.-
when
()¶ Повертає запланований час зворотного виклику як
float
секунди.Час – це абсолютна позначка часу, яка використовує те саме посилання на час, що й
loop.time()
.Нове в версії 3.7.
-
Серверні об’єкти¶
Серверні об’єкти створюються функціями loop.create_server()
, loop.create_unix_server()
, start_server()
і start_unix_server()
.
Do not instantiate the 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: Серверний об’єкт — це асинхронний менеджер контексту, починаючи з Python 3.7.
-
close
()¶ Зупинити обслуговування: закрийте сокети прослуховування та встановіть для атрибута
sockets
значенняNone
.Сокети, які представляють наявні вхідні підключення клієнта, залишаються відкритими.
The server is closed asynchronously, use the
wait_closed()
coroutine to wait until the server is closed.
-
get_loop
()¶ Повертає цикл подій, пов’язаний з об’єктом сервера.
Нове в версії 3.7.
-
coroutine
start_serving
()¶ Почніть приймати підключення.
This method is idempotent, so it can be called when the server is already being serving.
Ключовий параметр start_serving для
loop.create_server()
іasyncio.start_server()
дозволяє створити об’єкт сервера, який спочатку не приймає підключення. У цьому випадкуServer.start_serving()
абоServer.serve_forever()
можна використати, щоб сервер почав приймати з’єднання.Нове в версії 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.
-
is_serving
()¶ Повертає
True
, якщо сервер приймає нові підключення.Нове в версії 3.7.
-
sockets
¶ List of
socket.socket
objects the server is listening on.Змінено в версії 3.7: До Python 3.7
Server.sockets
використовувався для безпосереднього повернення внутрішнього списку серверних сокетів. У 3.7 повертається копія цього списку.
-
Реалізації циклу подій¶
asyncio постачається з двома різними реалізаціями циклу подій: SelectorEventLoop
і ProactorEventLoop
.
By default asyncio is configured to use SelectorEventLoop
on Unix and ProactorEventLoop
on Windows.
-
class
asyncio.
SelectorEventLoop
¶ An event loop based on the
selectors
module.Використовує найефективніший селектор, доступний для даної платформи. Також можна вручну налаштувати точну реалізацію селектора, яка буде використовуватися:
import asyncio import selectors selector = selectors.SelectSelector() loop = asyncio.SelectorEventLoop(selector) asyncio.set_event_loop(loop)
Наявність: Unix, Windows.
-
class
asyncio.
ProactorEventLoop
¶ An event loop for Windows that uses «I/O Completion Ports» (IOCP).
Наявність: Windows.
-
class
asyncio.
AbstractEventLoop
¶ Абстрактний базовий клас для асинційно-сумісних циклів подій.
The Event Loop Methods section lists all methods that an alternative implementation of
AbstractEventLoop
should have defined.
Приклади¶
Зверніть увагу, що всі приклади в цьому розділі цілеспрямовано показують, як використовувати API циклу подій низького рівня, такі як loop.run_forever()
і loop.call_soon()
. Сучасні асинхронні програми рідко потрібно писати таким чином; подумайте про використання функцій високого рівня, таких як asyncio.run()
.
Привіт, світ із call_soon()¶
Приклад використання методу 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.get_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.get_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.get_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¶
(Цей приклад сигналів
працює лише в Unix.)
Register handlers for signals SIGINT
and SIGTERM
using the loop.add_signal_handler()
method:
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())