« Контекстные менеджеры в Python — это удивительный механизм, который позволяет гарантировать корректное управление ресурсами и обеспечивать безопасное выполнение кода.» — Гвидо ван Россум, создатель языка программирования Python.

Что же удивительного в в этом механизме?

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

Итак, приступим к открытию файла, делается это просто. Как это делают новички:

file = open("file.txt", "r")
try:
    # Действия с файлом
    content = file.read()
    print(content)
finally:
    file.close()

В этом примере мы используем функцию open() для открытия файла "file.txt" в режиме чтения ("r"). Затем мы выполняем необходимые операции с файлом, в данном случае читаем его содержимое и выводим на экран. В блоке finally мы закрываем файл, чтобы освободить системные ресурсы, даже в случае возникновения исключений. Уже лучше, но сколько лишнего кода! Однако, код ниже является рекомендуемым.

В обоих случаях можно ненароком забыть о close(). Проблема решается простым использованием оператора with:

with open("file.txt", "r") as file:
    content = file.read()
    print(content)

После чтения файла автоматически будет вызван метод close(). Как все просто, правда?

Использование контекстного менеджера в других областях

Использование контекстного менеджера в Python для работы с базами данных (БД) и потоками может облегчить управление ресурсами и обеспечить безопасное выполнение операций. Ниже приведены примеры использования контекстных менеджеров для работы с БД и потоками в Python.

Работа с БД:

import sqlite3

# Пример работы с SQLite базой данных
with sqlite3.connect('example.db') as conn:
    cursor = conn.cursor()
    # Выполнение операций с базой данных
    cursor.execute('SELECT * FROM table_name')
    result = cursor.fetchall()
    print(result)

В этом примере контекстный менеджер sqlite3.connect() используется для установления соединения с SQLite базой данных. Затем выполняются операции с базой данных с использованием курсора. После завершения блока with соединение будет автоматически закрыто, освобождая ресурсы.

Работа с потоками:

import threading

# Пример работы с потоками
def worker():
    print("Работник выполняет задачу...")

with threading.Lock() as lock:
    # Блокировка ресурса перед выполнением задачи
    lock.acquire()
    try:
        t = threading.Thread(target=worker)
        t.start()
    finally:
        # Освобождение ресурса после выполнения задачи
        lock.release()

В этом примере контекстный менеджер threading.Lock() используется для блокировки ресурса перед выполнением задачи в отдельном потоке. Выполнение кода в блоке try происходит внутри захваченной блокировки, а после завершения блока with ресурс (блокировка) будет автоматически освобожден.

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

Сетевые соединения:
Использование контекстного менеджера с сетевыми соединениями в Python обеспечивает удобство и безопасность при работе с сетевыми ресурсами. Вот пример использования контекстного менеджера для работы с сетевым соединением с использованием модуля socket:

import socket

with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
    s.connect(('example.com', 80))
    # Выполнение операций с сетевым соединением
    s.sendall(b'GET / HTTP/1.1\r\nHost: example.com\r\n\r\n')
    response = s.recv(1024)
    print(response.decode())

В этом примере контекстный менеджер socket.socket() используется для создания сокета для TCP-соединения (socket.SOCK_STREAM) с использованием семейства адресов IPv4 (socket.AF_INET). Затем, в блоке with, мы устанавливаем соединение с удаленным хостом, отправляем HTTP-запрос и получаем ответ от сервера. После завершения блока with сокет будет автоматически закрыт, освобождая ресурсы.

Это простой пример, но контекстные менеджеры могут также использоваться для более сложных сценариев работы с сетевыми соединениями, такими как установка защищенного соединения с помощью SSL/TLS или использование протокола UDP.

Использование контекстного менеджера для работы с сетевыми соединениями облегчает управление ресурсами, обеспечивает надежность и гарантирует корректное закрытие соединения после использования, что особенно важно для предотвращения утечек ресурсов.

Как реализовать свой контекстный менеджер

Чтобы создать свой контекстный менеджер в Python, вам необходимо определить класс, который содержит методы __enter__() и __exit__().

Метод __enter__() выполняется перед выполнением блока кода внутри оператора with. Он может выполнять какие-либо подготовительные действия или возвращать значение, которое будет связано с переменной после ключевого слова as.

Метод __exit__() вызывается после завершения выполнения блока кода with. Он используется для выполнения завершающих действий, таких как освобождение ресурсов, обработка исключений или выполнение финализирующих операций.

Вот пример простого контекстного менеджера, который записывает время выполнения блока кода:

import time

class Timer:
    def __enter__(self):
        self.start_time = time.time()
        return self
    
    def __exit__(self, exc_type, exc_val, exc_tb):
        elapsed_time = time.time() - self.start_time
        print(f"Elapsed time: {elapsed_time} seconds")

# Пример использования контекстного менеджера
with Timer() as timer:
    # Ваш блок кода
    time.sleep(2)

Также можно даже реализовать асинхронный контекстный менеджер:

import asyncio

class AsyncTimer:
    def __enter__(self):
        self.start_time = asyncio.get_event_loop().time()
        return self
    
    def __aexit__(self, exc_type, exc_val, exc_tb):
        elapsed_time = asyncio.get_event_loop().time() - self.start_time
        print(f"Elapsed time: {elapsed_time} seconds")

# Пример использования асинхронного контекстного менеджера
async def example():
    async with AsyncTimer() as timer:
        # Ваш асинхронный блок кода
        await asyncio.sleep(2)

# Запуск асинхронной функции
asyncio.run(example())

Асинхронный контекстный менеджер используется в асинхронном программировании для управления ресурсами и выполнения финализирующих операций в асинхронных окружениях. Он предоставляет асинхронные версии методов __enter__() и __exit__(), которые могут быть использованы в блоке async with для автоматического управления ресурсами и обработки ошибок.

Вот несколько причин, почему асинхронный контекстный менеджер может быть полезным:

  1. Управление асинхронными ресурсами: Асинхронный контекстный менеджер позволяет управлять асинхронными ресурсами, такими как соединения с базами данных, сетевые соединения или файловые дескрипторы. Он обеспечивает автоматическое открытие и закрытие ресурсов, что упрощает их использование и предотвращает утечки ресурсов.

  2. Асинхронные операции пред- и пост-обработки: Асинхронный контекстный менеджер позволяет выполнить асинхронные операции перед выполнением блока кода async with (в методе __enter__()), а также после завершения блока (в методе __aexit__()). Это может быть полезно для инициализации, очистки или финализации ресурсов, асинхронных вычислений или обработки ошибок.

  3. Обработка исключений: Асинхронный контекстный менеджер предоставляет механизм для обработки исключений, возникающих в асинхронном блоке кода. Метод __aexit__() может быть использован для ловли исключений, их обработки и выполнения соответствующих действий, например, для отката транзакции или восстановления состояния.

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

Заключение

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

Комментарии (6)


  1. igorzakhar
    01.06.2023 19:14
    +6

    Сюда можно добавить про библиотеку contextlib с декоратором contextmanager.

    Нет необходимости писать класс для нового контекстного менеджера, достаточно обернуть генератор в декоратор @contextmanager. Ссылка на доку: contextlib — Utilities for with-statement contexts

    Пример из документации:

    from contextlib import contextmanager
    
    @contextmanager
    def managed_resource(*args, **kwds):
        # Code to acquire resource, e.g.:
        resource = acquire_resource(*args, **kwds)
        try:
            yield resource
        finally:
            # Code to release resource, e.g.:
            release_resource(resource)
    

    Использование:

    with managed_resource(timeout=3600) as resource:
        # Resource is released at the end of this block,
        # even if code in the block raises an exception

    Там много ещё чего.


    1. freeam Автор
      01.06.2023 19:14

      Я думаю это можно вообще в отдельную статью можно вынести.


  1. ammo
    01.06.2023 19:14

    class AsyncTimer:

    def __enter__(self):

    async def __aexit__(self, exc_type, exc_val, exc_tb):


    С такой комбинацией дандеров это в принципе не контекст менеджер.


    1. freeam Автор
      01.06.2023 19:14

      Спасибо, поправил.


  1. Buchachalo
    01.06.2023 19:14

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


  1. omaxx
    01.06.2023 19:14

    А вы не могли бы более подробно объяснить, что делает вот этот код?

    with threading.Lock() as lock:
        # Блокировка ресурса перед выполнением задачи
        lock.acquire()
        try:
            t = threading.Thread(target=worker)
            t.start()
        finally:
            # Освобождение ресурса после выполнения задачи
            lock.release()