Контекстный менеджер with в Python — как работает и зачем нужен

19 июня 2026
Автор

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

Введение

Когда ты работаешь с файлами, сетевыми соединениями или блокировками в Python, нужно всегда помнить о правильном освобождении ресурсов. Если забыть закрыть файл или разорвать соединение — программа может работать некорректно, утечь память или заблокировать данные. Именно для этого в Python существует оператор with и контекстные менеджеры — они автоматически берут на себя управление ресурсами.

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

Проблема без контекстного менеджера

Представь, что ты открываешь файл для чтения:

# Открываем файл вручную
file = open("data.txt", "r")
content = file.read()
print(content)
file.close()  # Не забыть закрыть!

Всё работает, пока между open() и close() не произойдёт ошибка. Если исключение вылетит до file.close(), файл останется открытым:

file = open("data.txt", "r")
content = file.read()
result = int(content)  # Если в файле не число — будет ValueError
file.close()  # Эта строка не выполнится при ошибке

Можно обернуть в try/finally:

file = open("data.txt", "r")
try:
    content = file.read()
    result = int(content)
finally:
    file.close()  # Закроется в любом случае

Но это громоздко. Оператор with решает эту задачу элегантнее.

Как работает оператор with

Оператор with гарантирует, что ресурс будет корректно освобождён независимо от того, произошла ошибка или нет:

# Файл закроется автоматически после выхода из блока with
with open("data.txt", "r") as file:
    content = file.read()
    result = int(content)
# Здесь file уже закрыт — даже если было исключение

Под капотом with вызывает два специальных метода объекта:

  • __enter__() — вызывается при входе в блок with. Возвращает объект, который присваивается переменной после as
  • __exit__() — вызывается при выходе из блока, даже если произошло исключение

Вот что происходит шаг за шагом:

# Это:
with open("data.txt") as f:
    data = f.read()

# Эквивалентно этому:
manager = open("data.txt")
f = manager.__enter__()
try:
    data = f.read()
finally:
    manager.__exit__(None, None, None)

Работа с файлами

Самый частый случай использования with — работа с файлами:

# Чтение файла
with open("config.json", "r", encoding="utf-8") as f:
    config = f.read()
    print(config)

# Запись в файл
with open("output.txt", "w", encoding="utf-8") as f:
    f.write("Привет, мир!\n")
    f.write("Вторая строка\n")

# Дозапись в файл
with open("log.txt", "a", encoding="utf-8") as f:
    f.write("Новая запись в лог\n")

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

Работа с несколькими ресурсами

Можно открывать несколько ресурсов в одном операторе with:

# Копируем содержимое одного файла в другой
with open("source.txt", "r") as src, open("dest.txt", "w") as dst:
    for line in src:
        dst.write(line.upper())

Начиная с Python 3.10 можно использовать скобки для переноса на несколько строк:

with (
    open("source.txt", "r") as src,
    open("dest.txt", "w") as dst,
    open("log.txt", "a") as log,
):
    for line in src:
        dst.write(line.upper())
        log.write(f"Обработана строка: {line.strip()}\n")

Работа с блокировками (threading)

Контекстные менеджеры незаменимы при работе с многопоточностью:

import threading

lock = threading.Lock()

# Без with — легко забыть отпустить блокировку
lock.acquire()
try:
    # Критическая секция
    shared_data += 1
finally:
    lock.release()

# С with — блокировка освобождается автоматически
with lock:
    # Критическая секция
    shared_data += 1

Работа с базами данных

Многие библиотеки для баз данных поддерживают контекстные менеджеры:

import sqlite3

# Соединение закроется автоматически
with sqlite3.connect("mydb.sqlite") as conn:
    cursor = conn.cursor()
    cursor.execute("SELECT * FROM users")
    rows = cursor.fetchall()
    for row in rows:
        print(row)
# При успешном выходе — автоматический commit
# При исключении — автоматический rollback

Вложенные контекстные менеджеры

Контекстные менеджеры можно вкладывать друг в друга:

import json

with open("config.json", "r", encoding="utf-8") as f:
    config = json.load(f)
    db_path = config["database"]

    with sqlite3.connect(db_path) as conn:
        cursor = conn.cursor()
        cursor.execute("SELECT count(*) FROM users")
        count = cursor.fetchone()[0]
        print(f"Всего пользователей: {count}")

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

  • Использование переменной после выхода из блока with. После закрытия файла обращение к нему вызовет ValueError: I/O operation on closed file:
with open("data.txt") as f:
    pass
f.read()  # Ошибка — файл уже закрыт
  • Забыть ключевое слово as. Без as контекстный менеджер работает, но ты не получишь ссылку на объект:
with open("data.txt"):
    pass  # Файл откроется и закроется, но прочитать его не получится
  • Путать with и обычное присваивание. Оператор with — это не просто присваивание, а управление жизненным циклом ресурса. Нельзя написать f = with open(...).

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

Можно ли использовать with без as?

Да. Если тебе не нужна ссылка на объект, можно написать with some_manager(): без as. Это полезно, например, при подавлении исключений или измерении времени.

Что происходит при исключении внутри блока with?

Метод __exit__() всё равно вызывается. Он получает информацию об исключении и может его подавить, вернув True. Если __exit__() возвращает False или None — исключение продолжит распространяться.

Работает ли with с любыми объектами?

Нет, только с объектами, у которых определены методы __enter__() и __exit__(). Такие объекты называются контекстными менеджерами. Большинство стандартных ресурсов Python (файлы, сокеты, блокировки) уже поддерживают этот протокол.

Заключение

Оператор with — один из самых полезных инструментов Python для безопасной работы с ресурсами. Он гарантирует корректное освобождение ресурсов, делает код чище и защищает от утечек. Используй with везде, где работаешь с файлами, соединениями, блокировками и другими ресурсами, требующими явного закрытия.

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

Стрелочка влевоМножественное наследование в Python — примеры и MROКомментарии в Python — однострочные, многострочные и docstringСтрелочка вправо

Постройте личный план изучения 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 ₽
Подробнее

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