Модуль contextlib в Python — утилиты для контекстных менеджеров

19 июня 2026
Автор

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

Введение

Модуль contextlib из стандартной библиотеки Python предоставляет набор утилит для работы с контекстными менеджерами. Вместо того чтобы каждый раз писать класс с __enter__ и __exit__, ты можешь создать контекстный менеджер буквально в несколько строк с помощью декоратора @contextmanager. А ещё в модуле есть готовые инструменты для подавления исключений, перенаправления вывода и управления стеком ресурсов.

В этой статье разберём основные инструменты contextlib и покажем, как применять их на практике.

Декоратор @contextmanager

Декоратор @contextmanager позволяет создать контекстный менеджер из обычной генераторной функции. Код до yield выполняется при входе в блок with, код после yield — при выходе:

from contextlib import contextmanager

@contextmanager
def timer():
    import time
    start = time.perf_counter()
    yield  # Здесь выполняется тело блока with
    elapsed = time.perf_counter() - start
    print(f"Время выполнения: {elapsed:.4f} сек")

# Использование
with timer():
    total = sum(range(10_000_000))
    print(f"Результат: {total}")

Если нужно передать значение в переменную as, используй yield с аргументом:

@contextmanager
def open_file(path, mode="r"):
    f = open(path, mode, encoding="utf-8")
    try:
        yield f  # Это значение попадёт в переменную после as
    finally:
        f.close()

with open_file("data.txt", "w") as f:
    f.write("Привет из contextmanager!")

Обработка исключений в @contextmanager

Блок try/finally внутри генератора обязателен, если нужно гарантировать выполнение финализации:

@contextmanager
def managed_resource(name):
    print(f"Захватываем ресурс: {name}")
    try:
        yield name
    except Exception as e:
        print(f"Ошибка при работе с {name}: {e}")
        raise  # Пробрасываем исключение дальше
    finally:
        print(f"Освобождаем ресурс: {name}")

with managed_resource("база данных") as res:
    print(f"Работаем с: {res}")
    # Даже при ошибке ресурс будет освобождён

contextlib.suppress — подавление исключений

Функция suppress() создаёт контекстный менеджер, который подавляет указанные типы исключений:

from contextlib import suppress

# Без suppress
try:
    os.remove("temp.txt")
except FileNotFoundError:
    pass

# С suppress — то же самое, но короче
with suppress(FileNotFoundError):
    os.remove("temp.txt")

Можно подавлять несколько типов исключений:

with suppress(FileNotFoundError, PermissionError):
    os.remove("protected_file.txt")

print("Программа продолжает работу")

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

contextlib.closing — оборачивание объектов без поддержки with

Если объект имеет метод close(), но не реализует протокол контекстного менеджера, используй closing():

from contextlib import closing
from urllib.request import urlopen

# urlopen возвращает объект с методом close(), но без __enter__/__exit__
with closing(urlopen("https://example.com")) as page:
    content = page.read()
    print(f"Получено {len(content)} байт")
# page.close() вызовется автоматически

contextlib.redirectstdout и redirectstderr

Эти менеджеры перенаправляют стандартный вывод или вывод ошибок:

from contextlib import redirect_stdout, redirect_stderr
import io

# Перехватываем stdout в строку
f = io.StringIO()
with redirect_stdout(f):
    print("Это не появится в консоли")
    print("И это тоже")

output = f.getvalue()
print(f"Перехваченный вывод: {output!r}")

Полезно для тестирования функций, которые печатают в консоль:

def greet(name):
    print(f"Привет, {name}!")

# Тестируем вывод функции
buffer = io.StringIO()
with redirect_stdout(buffer):
    greet("Алиса")

assert buffer.getvalue() == "Привет, Алиса!\n"

Перенаправление stderr:

err_buffer = io.StringIO()
with redirect_stderr(err_buffer):
    import warnings
    warnings.warn("Тестовое предупреждение")

print(f"Перехваченные ошибки: {err_buffer.getvalue()}")

contextlib.ExitStack — динамический стек ресурсов

ExitStack позволяет управлять произвольным количеством контекстных менеджеров, которые определяются во время выполнения:

from contextlib import ExitStack

# Открываем произвольное количество файлов
filenames = ["file1.txt", "file2.txt", "file3.txt"]

with ExitStack() as stack:
    files = [
        stack.enter_context(open(fname, "w", encoding="utf-8"))
        for fname in filenames
    ]
    # Все файлы открыты
    for i, f in enumerate(files):
        f.write(f"Содержимое файла {i + 1}\n")
# Все файлы автоматически закрыты

Регистрация callback-функций

ExitStack умеет регистрировать произвольные функции для вызова при выходе:

from contextlib import ExitStack

def cleanup(name):
    print(f"Очистка: {name}")

with ExitStack() as stack:
    stack.callback(cleanup, "временные файлы")
    stack.callback(cleanup, "кэш")
    stack.callback(cleanup, "соединения")
    print("Выполняем основную работу...")

# Вывод (в обратном порядке — как стек):
# Выполняем основную работу...
# Очистка: соединения
# Очистка: кэш
# Очистка: временные файлы

Передача владения ресурсами

Метод pop_all() позволяет «забрать» ресурсы из стека, чтобы они не закрылись автоматически:

from contextlib import ExitStack

def open_files_safely(filenames):
    with ExitStack() as stack:
        files = []
        for fname in filenames:
            f = stack.enter_context(open(fname, "r", encoding="utf-8"))
            files.append(f)
        # Если всё успешно — забираем ответственность за закрытие
        return stack.pop_all(), files
    # Если произошла ошибка — все ранее открытые файлы закроются

# Вызывающий код сам закрывает файлы
closer, files = open_files_safely(["data1.txt", "data2.txt"])
try:
    for f in files:
        print(f.read())
finally:
    closer.close()

contextlib.nullcontext — заглушка

nullcontext — контекстный менеджер, который ничего не делает. Полезен для условных конструкций:

from contextlib import nullcontext

def process_data(data, logfile=None):
    # Если указан лог-файл — открываем, иначе используем заглушку
    cm = open(logfile, "w", encoding="utf-8") if logfile else nullcontext()

    with cm as f:
        for item in data:
            result = item * 2
            if f is not None:
                f.write(f"{item} -> {result}\n")

process_data([1, 2, 3], logfile="log.txt")  # С логированием
process_data([4, 5, 6])  # Без логирования

Асинхронные контекстные менеджеры

Модуль contextlib поддерживает и асинхронные варианты:

from contextlib import asynccontextmanager
import asyncio

@asynccontextmanager
async def async_timer():
    import time
    start = time.perf_counter()
    try:
        yield
    finally:
        elapsed = time.perf_counter() - start
        print(f"Async время: {elapsed:.4f} сек")

async def main():
    async with async_timer():
        await asyncio.sleep(1)
        print("Работа завершена")

asyncio.run(main())

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

  • Забыть try/finally в @contextmanager. Без try/finally код после yield не выполнится при исключении:
# Плохо — cleanup не вызовется при ошибке
@contextmanager
def bad_manager():
    print("setup")
    yield
    print("cleanup")  # Не выполнится при исключении!

# Правильно
@contextmanager
def good_manager():
    print("setup")
    try:
        yield
    finally:
        print("cleanup")  # Выполнится всегда
  • Делать yield больше одного раза. В @contextmanager допускается ровно один yield. Второй вызовет RuntimeError.

  • Использовать suppress для слишком широких исключений. suppress(Exception) подавит практически все ошибки, что затруднит отладку.

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

Когда использовать @contextmanager, а когда класс?

Если логика простая (настроить — сделать — убрать) — используй @contextmanager. Если нужна сложная обработка исключений, хранение состояния между вызовами или наследование — используй класс.

Можно ли комбинировать suppress и ExitStack?

Да. ExitStack принимает любые контекстные менеджеры, включая suppress:

with ExitStack() as stack:
    stack.enter_context(suppress(FileNotFoundError))
    os.remove("maybe_missing.txt")

Чем nullcontext отличается от suppress?

nullcontext — это «пустой» менеджер, который вообще ничего не делает. suppress активно перехватывает и подавляет указанные исключения. Это разные инструменты для разных задач.

Заключение

Модуль contextlib значительно упрощает работу с контекстными менеджерами в Python. Декоратор @contextmanager позволяет создать менеджер в пару строк, suppress заменяет шаблонные try/except/pass, а ExitStack решает задачу управления динамическим набором ресурсов. Эти инструменты делают код чище и надёжнее.

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

Стрелочка влевоИспользование Anaconda с 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 выбрать для установкиКак вывести целое число с помощью 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 ₽
Подробнее

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