Продвинутые генераторы в Python — send, throw, close и корутины

19 июня 2026
Автор

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

Введение

Генераторы в Python — это не только ленивые итераторы. Благодаря методам send(), throw() и close() они становятся полноценными корутинами, способными принимать данные извне, обрабатывать исключения и управлять своим жизненным циклом. Эти возможности лежат в основе асинхронного программирования и многих продвинутых паттернов.

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

Метод send() — отправка данных в генератор

Метод send() позволяет не только получать значения из генератора, но и передавать данные обратно. Значение, переданное через send(), становится результатом выражения yield внутри генератора:

def accumulator():
    """Генератор-аккумулятор: принимает числа и возвращает текущую сумму"""
    total = 0
    while True:
        # yield возвращает total и принимает новое значение в value
        value = yield total
        if value is not None:
            total += value

acc = accumulator()
# Первый вызов — инициализация генератора (send(None) или next())
next(acc)         # 0 — начальное значение total

print(acc.send(10))  # 10
print(acc.send(20))  # 30
print(acc.send(5))   # 35

Важно: первый вызов генератора должен быть next() или send(None), чтобы выполнение дошло до первого yield. Попытка отправить значение в «неинициализированный» генератор вызовет TypeError.

Паттерн корутины

Корутина — это генератор, который в основном принимает данные через send(), а не генерирует последовательность. Типичная корутина использует бесконечный цикл с yield:

def running_average():
    """Корутина для вычисления скользящего среднего"""
    total = 0.0
    count = 0
    average = None
    while True:
        value = yield average
        total += value
        count += 1
        average = total / count

# Используем корутину
avg = running_average()
next(avg)  # Инициализация

print(avg.send(10))   # 10.0
print(avg.send(20))   # 15.0
print(avg.send(30))   # 20.0
print(avg.send(40))   # 25.0

Декоратор для автоматической инициализации

Чтобы не вызывать next() вручную, можно создать декоратор:

from functools import wraps

def coroutine(func):
    """Декоратор для автоматической инициализации корутины"""
    @wraps(func)
    def wrapper(*args, **kwargs):
        gen = func(*args, **kwargs)
        next(gen)  # Инициализируем генератор
        return gen
    return wrapper

@coroutine
def filter_values(predicate):
    """Корутина-фильтр: пропускает только подходящие значения"""
    results = []
    while True:
        value = yield results
        if predicate(value):
            results.append(value)

# Фильтруем только положительные числа
f = filter_values(lambda x: x > 0)
# Не нужен вызов next() — декоратор сделал это за нас
print(f.send(5))    # [5]
print(f.send(-3))   # [5]
print(f.send(10))   # [5, 10]

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

Метод throw() — отправка исключений в генератор

Метод throw() позволяет выбросить исключение внутри генератора в точке yield. Генератор может обработать это исключение или пропустить его дальше:

def resilient_processor():
    """Генератор с обработкой ошибок"""
    while True:
        try:
            value = yield
            print(f"Обработано: {value}")
        except ValueError as e:
            print(f"Ошибка значения: {e}")
        except GeneratorExit:
            print("Генератор завершён")
            return

proc = resilient_processor()
next(proc)

proc.send("данные")       # Обработано: данные
proc.throw(ValueError, "неверный формат")  # Ошибка значения: неверный формат
proc.send("ещё данные")   # Обработано: ещё данные

Метод close() — завершение генератора

Метод close() выбрасывает исключение GeneratorExit внутри генератора, что приводит к его завершению:

def resource_manager():
    """Генератор, управляющий ресурсом"""
    print("Ресурс открыт")
    try:
        while True:
            data = yield
            print(f"Записано: {data}")
    except GeneratorExit:
        print("Ресурс закрыт — очистка завершена")

rm = resource_manager()
next(rm)            # Ресурс открыт
rm.send("строка 1") # Записано: строка 1
rm.send("строка 2") # Записано: строка 2
rm.close()          # Ресурс закрыт — очистка завершена

Метод close() полезен для корректного освобождения ресурсов: закрытия файлов, сетевых соединений и других объектов.

Генератор как конечный автомат

Генераторы с send() отлично подходят для реализации конечных автоматов:

def traffic_light():
    """Генератор-светофор: переключает состояния по сигналу"""
    state = "красный"
    transitions = {
        "красный": "зелёный",
        "зелёный": "жёлтый",
        "жёлтый": "красный"
    }
    while True:
        command = yield state
        if command == "next":
            state = transitions[state]
        elif command == "reset":
            state = "красный"

light = traffic_light()
print(next(light))          # красный
print(light.send("next"))   # зелёный
print(light.send("next"))   # жёлтый
print(light.send("next"))   # красный
print(light.send("next"))   # зелёный
print(light.send("reset"))  # красный

yield from и двусторонний обмен

Конструкция yield from не просто делегирует итерацию — она пробрасывает send() и throw() к вложенному генератору:

def inner():
    """Вложенный генератор, принимающий данные"""
    result = []
    while True:
        value = yield
        if value is None:
            return result  # Возвращает результат через StopIteration
        result.append(value * 2)

def outer():
    """Внешний генератор, делегирующий работу"""
    # yield from пробрасывает send() и собирает return
    result = yield from inner()
    yield result

gen = outer()
next(gen)
gen.send(1)
gen.send(2)
gen.send(3)
print(gen.send(None))  # [2, 4, 6]

Генераторы и itertools

Модуль itertools предоставляет инструменты, которые хорошо сочетаются с генераторами:

import itertools

def sensor_readings():
    """Имитация бесконечного потока показаний датчика"""
    import random
    while True:
        yield random.uniform(20.0, 30.0)

# Берём первые 5 показаний
readings = itertools.islice(sensor_readings(), 5)
for r in readings:
    print(f"{r:.1f}°C")

# Группировка по условию
data = [1, 1, 2, 3, 3, 3, 4, 5, 5]
for key, group in itertools.groupby(data):
    print(f"{key}: {list(group)}")

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

  • Вызов send() без инициализации. Перед первым send() с ненулевым аргументом нужно вызвать next() или send(None).
  • Игнорирование GeneratorExit в try/finally. Если генератор содержит try/finally, блок finally выполнится при close(). Убедитесь, что он не выбрасывает исключения.
  • Бесконечный генератор без close(). Если генератор управляет ресурсами, всегда завершайте его через close() или используйте контекстный менеджер.

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

Чем корутина на генераторах отличается от async/await? Корутины на генераторах — предшественник async/await. Они работают через yield и send(), а async/await — через специальный синтаксис, введённый в Python 3.5. Современный асинхронный код лучше писать через async/await, но понимание генераторных корутин помогает разобраться в механизмах Python.

Когда использовать send() вместо аргумента функции? Используйте send(), когда генератор должен сохранять состояние между вызовами и принимать данные порционно. Это полезно для потоковой обработки, конечных автоматов и конвейеров данных.

Безопасно ли вызывать close() на уже завершённом генераторе? Да, повторный вызов close() на завершённом генераторе не вызывает ошибок.

Заключение

Продвинутые возможности генераторов — send(), throw(), close() и yield from — превращают их из простых итераторов в мощный инструмент для управления потоком выполнения. Корутины на генераторах, конечные автоматы и двусторонний обмен данными — всё это доступно благодаря протоколу генераторов Python.

Для закрепления навыков работы с генераторами и освоения асинхронного программирования рекомендуем курс Python-разработчик с нуля. В первых 3 модулях курса доступно бесплатное содержание, что позволяет разобраться с основами языка и понять структуру курса до покупки полного доступа.

Стрелочка влевоКак выполняется вызов функций call в PythonПозиционные и именованные аргументы в 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 выбрать для установкиКак установить Python на Windows macOS и LinuxКак вывести целое число с помощью print в PythonКак пользоваться консолью 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 и вложенные шаблоныЛокальные и глобальные переменные в PythonПрактические примеры match/case в 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 ₽
Подробнее

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