Модуль asyncio в Python — основы асинхронного программирования

19 июня 2026
Автор

Антон Ларичев

Введение

Модуль asyncio — это стандартная библиотека Python для написания асинхронного кода. Он предоставляет событийный цикл (event loop), корутины, задачи и примитивы синхронизации, которые позволяют эффективно обрабатывать операции ввода-вывода без блокировки основного потока выполнения.

В этой статье мы разберём ключевые компоненты asyncio, научимся создавать и запускать корутины, управлять задачами и использовать событийный цикл для построения асинхронных приложений.

Что такое событийный цикл

Событийный цикл (event loop) — это центральный механизм asyncio. Он управляет выполнением корутин, обрабатывает операции ввода-вывода и планирует обратные вызовы. Можно представить его как диспетчер, который решает, какая задача должна выполняться в данный момент.

import asyncio

async def main():
    print("Привет из событийного цикла!")

# Запуск событийного цикла
asyncio.run(main())

Функция asyncio.run() — это точка входа в асинхронный код. Она создаёт новый событийный цикл, запускает переданную корутину и закрывает цикл после её завершения.

Корутины и их создание

Корутина — это специальная функция, объявленная с ключевым словом async def. В отличие от обычной функции, корутина не выполняется сразу при вызове — она возвращает объект корутины, который нужно запустить через событийный цикл.

import asyncio

async def privetstvie(imya):
    # Имитируем асинхронную задержку
    await asyncio.sleep(1)
    return f"Привет, {imya}!"

async def main():
    # Вызов корутины с ожиданием результата
    rezultat = await privetstvie("Мир")
    print(rezultat)

asyncio.run(main())

Ключевое слово await приостанавливает выполнение текущей корутины до тех пор, пока ожидаемая операция не завершится. В это время событийный цикл может выполнять другие задачи.

Задачи (Tasks)

Задачи позволяют запускать несколько корутин параллельно. Когда вы оборачиваете корутину в задачу, она начинает выполняться в фоне, не блокируя остальной код.

import asyncio

async def zagruzka(url, zaderjka):
    # Имитируем загрузку данных
    print(f"Начинаю загрузку {url}...")
    await asyncio.sleep(zaderjka)
    print(f"Загрузка {url} завершена!")
    return f"Данные с {url}"

async def main():
    # Создаём задачи для параллельного выполнения
    zadacha1 = asyncio.create_task(zagruzka("api/users", 2))
    zadacha2 = asyncio.create_task(zagruzka("api/posts", 1))
    zadacha3 = asyncio.create_task(zagruzka("api/comments", 3))

    # Ожидаем завершения всех задач
    rezultat1 = await zadacha1
    rezultat2 = await zadacha2
    rezultat3 = await zadacha3

    print(f"Получено: {rezultat1}, {rezultat2}, {rezultat3}")

asyncio.run(main())

В этом примере три загрузки выполняются параллельно, и общее время выполнения составит около 3 секунд (по самой долгой задаче), а не 6 секунд при последовательном выполнении.

Если вы хотите детальнее изучить асинхронное программирование в Python — приходите на наш большой курс Продвинутый Python. На курсе 171 урок и 22 упражнения, AI-тренажёры для практики 24/7, решение задач с живым ревью наставника, еженедельные встречи с менторами.

Функция asyncio.gather

Функция asyncio.gather() — удобный способ запустить несколько корутин одновременно и собрать все результаты в один список.

import asyncio

async def poluchit_dannye(istochnik, zaderjka):
    await asyncio.sleep(zaderjka)
    return f"Данные из {istochnik}"

async def main():
    # Запускаем три корутины параллельно
    rezultaty = await asyncio.gather(
        poluchit_dannye("база данных", 2),
        poluchit_dannye("внешний API", 3),
        poluchit_dannye("кэш", 1),
    )

    for rezultat in rezultaty:
        print(rezultat)

asyncio.run(main())

Результаты возвращаются в том же порядке, в котором были переданы корутины, независимо от того, какая из них завершилась первой.

Таймауты с asyncio.wait_for

Иногда нужно ограничить время ожидания операции. Для этого используется asyncio.wait_for():

import asyncio

async def dolgaya_operaciya():
    # Операция, которая выполняется 10 секунд
    await asyncio.sleep(10)
    return "Готово"

async def main():
    try:
        # Устанавливаем таймаут в 3 секунды
        rezultat = await asyncio.wait_for(dolgaya_operaciya(), timeout=3.0)
        print(rezultat)
    except asyncio.TimeoutError:
        print("Операция превысила лимит времени!")

asyncio.run(main())

Асинхронные генераторы и итераторы

Python поддерживает асинхронные версии генераторов и итераторов с помощью async for:

import asyncio

async def asinhronniy_generator(n):
    for i in range(n):
        # Имитируем асинхронное получение данных
        await asyncio.sleep(0.5)
        yield i * 2

async def main():
    async for znachenie in asinhronniy_generator(5):
        print(f"Получено значение: {znachenie}")

asyncio.run(main())

Частые ошибки

  • Забыть await перед вызовом корутины — без await корутина не выполнится, а вернёт объект корутины. Python выдаст предупреждение RuntimeWarning: coroutine was never awaited.
  • Вызов asyncio.run() внутри уже запущенного событийного цикла — это приведёт к ошибке RuntimeError. Вместо этого используйте await или asyncio.create_task().
  • Использование блокирующих операций в асинхронном коде — функции вроде time.sleep() или синхронные сетевые вызовы заблокируют весь событийный цикл. Используйте asyncio.sleep() и асинхронные библиотеки.
  • Не обрабатывать исключения в задачах — если задача выбросит исключение и оно не будет обработано, оно может быть потеряно до момента сборки мусора.

Частозадаваемые вопросы

Чем asyncio отличается от потоков (threading)? Asyncio использует кооперативную многозадачность в одном потоке — корутины сами уступают управление через await. Потоки используют вытесняющую многозадачность, где переключение между потоками происходит на уровне ОС. Asyncio лучше подходит для задач с большим количеством операций ввода-вывода, а потоки — для CPU-интенсивных задач.

Можно ли использовать asyncio с обычными синхронными функциями? Да, для этого используйте asyncio.to_thread() (Python 3.9+) или loop.run_in_executor(), чтобы запустить синхронную функцию в отдельном потоке без блокировки событийного цикла.

Когда стоит использовать asyncio.gather, а когда asyncio.create_task? asyncio.gather() удобен, когда вам нужно дождаться завершения группы корутин и получить все результаты. create_task() больше подходит для задач, которые запускаются в фоне и с которыми вы работаете по отдельности.

Заключение

Модуль asyncio — мощный инструмент для написания эффективного асинхронного кода в Python. Событийный цикл, корутины и задачи позволяют обрабатывать тысячи операций ввода-вывода без создания множества потоков. Для закрепления навыков асинхронного программирования рекомендуем курс Продвинутый Python. В первых 3 модулях курса доступно бесплатное содержание, что позволяет разобраться в основах и понять структуру курса до покупки полного доступа.

Синтаксис async/await в Python — как писать асинхронный кодСтрелочка вправо

Постройте личный план изучения Python до уровня Middle — бесплатно!

Python — часть карты развития Backend

  • step100+ шагов развития
  • lessons30 бесплатных лекций
  • lessons300 бонусных рублей на счет

Все гайды по Python

Как отправлять запросы с помощью requests в PythonПочему Python выводит значение без команды printКак работает команда print в PythonВозможности Python для автоматизации задачРабота с JSON в Python на примерахPython get — методы получения данныхКак находить и исправлять ошибки в PythonРабота с данными через API и внешние сервисыСтруктура и оформление кода PythonОсновы Django с PythonПолезные приёмы в Python для повседневной работыИспользование locals в Python для отладкиИнтеграция PHP и PythonКак выполнять HTTPS-запросы в PythonFastAPI Python — быстрый старт: создание REST API с нуляКак работать с API в Python
Ввод целого числа в PythonВедение логов в PythonУдаление данных в Python с помощью removeОбработка исключений с помощью try/except в PythonФункция super() в Python — как вызвать метод родителяСоздание собственных контекстных менеджеров в PythonРабота с символами программирования PythonРабота с переменной X в PythonРабота с классами в PythonКак скачать Python на компьютерПростая программа на Python для начинающихОсновы Python для тех, кто начинаетЧто нового в Python 3Поддерживается ли Python 2 и стоит ли его использоватьPython 1 — с чего начиналась история языкаКоманда python print - полное руководство по выводу данныхПравила именования переменных в PythonПользовательские исключения в PythonОсновы Python coreОписание объектов PythonНаследование классов в Python — основы и примерыМножественное наследование в Python — примеры и MROКонтекстный менеджер with в Python — как работает и зачем нуженКомментарии в Python — однострочные, многострочные и docstringКакой Python выбрать для установкиКак вывести целое число с помощью print в PythonКак установить Python на Windows macOS и LinuxКак пользоваться консолью PythonКак получить последний элемент в PythonКак найти значение в PythonКак настроить PythonКак использовать print для строк в PythonКак работает интерпретатор PythonИнструкция по работе с PythonБлок finally в обработке исключений PythonЦелые числа в PythonАбстрактные классы в Python — ABC и abstractmethod
Загрузка данных PythonУправление проектами на GitHub с PythonСоздание веб-приложений на Flask PythonСоздание бота на PythonСоздание интерфейсов Python QTСоздание игр с PygameСоздание GUI в PythonКак работать со словарями в PythonРабота с библиотеками через Python PackagingРабота со временем в Python при помощи модуля timePython name — особенности переменнойМатематические операции в Python с модулем mathPython listing — что это и как использоватьPytest — тестирование на Python: полное руководствоОбработка изображений с OpenCV PythonNumPy в Python — основы и применение в задачахМашинное обучение с PythonИспользование Anaconda с PythonМодуль contextlib в Python — утилиты для контекстных менеджеровБиблиотеки Python и их применение в проектах
Возврат значений из функции в PythonВложенные функции в PythonСоздание собственных декораторов в PythonРабота с функцией map в PythonЦикл while в Python и примеры использованияОбработка чисел, введённых через input в PythonОсновные операторы в Python с примерамиУсловные выражения if else в Python для начинающихКак выполняется вызов функций call в PythonПродвинутые генераторы в Python — send, throw, close и корутиныПозиционные и именованные аргументы в PythonОбъявление переменных и управление областью видимости в PythonПередача аргументов по ссылке и по значению в PythonПередача аргументов через args и kwargs в PythonОсновные методы Python и примеры их использованияОператор match/case в Python 3.10+ — основы структурного сопоставленияПаттерны match/case в Python — деструктуризация, guard и вложенные шаблоныПрактические примеры match/case в Python — реальные сценарии примененияЛокальные и глобальные переменные в PythonЧасто используемые команды PythonКлючевые слова global и nonlocal в PythonКак создавать функции в PythonКак работает сборщик мусора в PythonКак работает область видимости переменных в PythonКак работает функция callable в PythonКак работает функция any и all в PythonКак проверить тип переменной в PythonКак передать функцию как аргумент в PythonКак использовать функцию isinstance в PythonКак использовать функцию filter в PythonКак использовать функцию filter в PythonКак использовать функцию eval безопасно в PythonКак использовать декораторы в PythonИзменяемые и неизменяемые типы данных в PythonГенераторы и yield в Python — как создавать и использоватьГенераторные выражения в Python — синтаксис и примерыФункции в Python и способы их вызоваФункции как объекты в PythonЧто такое замыкания в PythonЧто делает функция reduce в PythonЧто делает функция id в PythonАргументы по умолчанию в PythonАнонимные функции и lambda в PythonАлгоритмы на Python — примеры и объяснение
Запись данных в PythonУстановка pip в PythonУправление зависимостями requirement в PythonУправление библиотеками с помощью Python PackagingУдаление пробелов с помощью strip в PythonСтруктурирование кода в PythonСоздание исполняемого файла Python в exeРазбор traceback в модуле PythonРазбор site-packages в PythonРазбор Program Files в PythonРабота с Unicode кодировками в PythonРабота с системными функциями Python sysРабота с папкой AppData в PythonРабота с модулем logging в PythonРабота с каталогами в PythonРабота с CSV в PythonВиртуальная среда venv в Python — создание и настройкаКак создать простое приложение на PythonИспользование pip в Python для установки пакетовМодули в Python и организация кода в проектеИмпорт модулей в Python и правила подключенияРабота с файлами в Python пошаговоЧто делает компилятор Python и как он работаетПолучение строки из модуля PythonПодключение файлов в Python с includeПеременные среды в PythonСборка проекта с помощью packaging в PythonНастройка Python сервераИспользование Python на UbuntuИспользование консоли PythonИспользование кодировок в PythonИнициализация пакетов PythonИмпорт модулей PythonИмпорт имен в PythonСреда IDLE Python и базовые возможностиЧтение и запись TXT в PythonЧтение файлов в Python с помощью open file
Удаление элементов из списка PythonТипы данных в Python — обзор и рекомендацииОсновные операции со строками в PythonМетоды str в Python и обработка текстаСписки в Python и их ключевые методыСоздание списков данных в PythonРабота со строками и символами в PythonРабота со столбцами в PythonРабота со списком значений в PythonРабота с таблицами в Python с помощью DataFrameРабота с RFR в PythonРабота с пробелами в PythonРабота с массивами в PythonРабота с кортежами tuple PythonРабота с координатами X и Y в PythonРабота с ключами в PythonРабота с элементами данных PythonРабота с двоичными числами PythonРабота с данными в PythonРабота с данными NumPy PythonРабота с большими числами в PythonРабота с битами в PythonРабота с байтами в PythonЧто такое значение в Python и как его определитьМножества в Python и операции с нимиИспользование range в Python для цикловПроверка на четность в PythonПроверка числа в PythonПреобразование типов в PythonПреобразование списка в строку PythonПреобразование числа в строку в PythonПостроение графиков в PythonОпределение индекса элемента в PythonОкругление чисел в PythonОбъединение списков в Python с помощью zipМножества в PythonМассивы в Python и отличие от списковМассив чисел в PythonКортежи данных в PythonКак вычислить сумму чисел в PythonКак получить остаток от деления в PythonКак найти следующее число в PythonИспользование Unicode в PythonТип int в Python и его особенностиИндекс списка в PythonФункции для работы со строками в PythonЭлементы Python и способы доступа к нимДоступ к элементам массива в PythonДеление чисел в PythonРабота с данными в Python на практикеКак работать с числами в Python
Открыть базу знаний

Лучшие курсы по теме

Иконка молнииНовый
изображение курса

Основы Python

Антон Ларичев
AI-тренажерыAI-тренажеры
Практика в студииПрактика в студии
Гарантия
Бонусы
иконка звёздочки рейтинга4.7
3 999 ₽ 6 990 ₽
Подробнее
изображение курса

Nest.js с нуля

Антон Ларичев
AI-тренажерыAI-тренажеры
Практика в студииПрактика в студии
Гарантия
Бонусы
иконка звёздочки рейтинга4.6
3 999 ₽ 6 990 ₽
Подробнее
изображение курса

Docker и Ansible

Антон Ларичев
AI-тренажерыAI-тренажеры
Гарантия
Бонусы
иконка звёздочки рейтинга4.7
3 999 ₽ 6 990 ₽
Подробнее

Отправить комментарий