Паттерны match/case в Python — деструктуризация, guard и вложенные шаблоны

19 июня 2026
Автор

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

Введение

Базовый match/case с литералами — это лишь вершина айсберга. Настоящая мощь структурного сопоставления в Python 3.10+ раскрывается при работе с паттернами: деструктуризацией кортежей и списков, guard-условиями, шаблонами классов и вложенными паттернами. Именно эти возможности отличают match/case от обычного switch в других языках.

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

Деструктуризация кортежей

Один из самых полезных сценариев — разбор кортежей на составные части прямо в шаблоне case.

# Деструктуризация кортежа с координатами
def describe_point(point):
    match point:
        case (0, 0):
            return "Начало координат"
        case (x, 0):
            return f"На оси X, x={x}"
        case (0, y):
            return f"На оси Y, y={y}"
        case (x, y):
            return f"Точка ({x}, {y})"

print(describe_point((0, 0)))    # Начало координат
print(describe_point((5, 0)))    # На оси X, x=5
print(describe_point((0, 3)))    # На оси Y, y=3
print(describe_point((2, 7)))    # Точка (2, 7)

Python автоматически распаковывает кортеж и привязывает каждый элемент к указанной переменной. Шаблон совпадает только если количество элементов и фиксированные значения (0 в данном случае) совпадают.

# Деструктуризация кортежа с тремя элементами
def handle_rgb(color):
    match color:
        case (0, 0, 0):
            return "Чёрный"
        case (255, 255, 255):
            return "Белый"
        case (r, 0, 0):
            return f"Оттенок красного (r={r})"
        case (0, g, 0):
            return f"Оттенок зелёного (g={g})"
        case (0, 0, b):
            return f"Оттенок синего (b={b})"
        case (r, g, b):
            return f"RGB({r}, {g}, {b})"

print(handle_rgb((255, 255, 255)))  # Белый
print(handle_rgb((128, 0, 0)))      # Оттенок красного (r=128)
print(handle_rgb((10, 20, 30)))     # RGB(10, 20, 30)

Деструктуризация списков и последовательностей

Аналогично кортежам можно деструктурировать списки. Также поддерживается оператор * для захвата оставшихся элементов.

# Деструктуризация списка с оператором *
def process_args(args):
    match args:
        case []:
            return "Нет аргументов"
        case [single]:
            return f"Один аргумент: {single}"
        case [first, second]:
            return f"Два аргумента: {first} и {second}"
        case [first, *rest]:
            return f"Первый: {first}, остальные: {rest}"

print(process_args([]))              # Нет аргументов
print(process_args(["hello"]))       # Один аргумент: hello
print(process_args(["a", "b"]))      # Два аргумента: a и b
print(process_args([1, 2, 3, 4]))    # Первый: 1, остальные: [2, 3, 4]
# Разбор команды с аргументами
def parse_command(tokens):
    match tokens:
        case ["quit"]:
            return "Выход из программы"
        case ["go", direction]:
            return f"Движение: {direction}"
        case ["drop", *items] if items:
            return f"Бросить предметы: {', '.join(items)}"
        case ["drop", *items]:
            return "Ошибка: укажите что бросить"
        case _:
            return "Неизвестная команда"

print(parse_command(["go", "north"]))           # Движение: north
print(parse_command(["drop", "sword", "shield"]))  # Бросить предметы: sword, shield
print(parse_command(["drop"]))                  # Ошибка: укажите что бросить

Guard-условия (if в case)

Guard-условие — это дополнительная проверка после шаблона, которая позволяет уточнить критерии совпадения. Блок case выполняется только если и шаблон совпал, и guard-условие вернуло True.

# Guard-условия для проверки диапазонов
def classify_temperature(temp):
    match temp:
        case t if t < -30:
            return "Экстремальный мороз"
        case t if t < 0:
            return "Мороз"
        case t if t < 10:
            return "Холодно"
        case t if t < 20:
            return "Прохладно"
        case t if t < 30:
            return "Тепло"
        case t:
            return "Жарко"

print(classify_temperature(-35))  # Экстремальный мороз
print(classify_temperature(15))   # Прохладно
print(classify_temperature(35))   # Жарко
# Комбинация деструктуризации и guard-условий
def process_order(order):
    match order:
        case (item, quantity) if quantity <= 0:
            return f"Ошибка: неверное количество для {item}"
        case (item, quantity) if quantity > 100:
            return f"Оптовый заказ: {item} x {quantity}"
        case (item, quantity):
            return f"Заказ: {item} x {quantity}"

print(process_order(("Книга", -1)))   # Ошибка: неверное количество для Книга
print(process_order(("Ручка", 500)))   # Оптовый заказ: Ручка x 500
print(process_order(("Тетрадь", 5)))   # Заказ: Тетрадь x 5

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

Деструктуризация словарей

match/case позволяет проверять наличие ключей в словаре и извлекать их значения. При этом словарь может содержать и другие ключи — они просто игнорируются.

# Деструктуризация словаря
def handle_event(event):
    match event:
        case {"type": "click", "x": x, "y": y}:
            return f"Клик в позиции ({x}, {y})"
        case {"type": "keypress", "key": key}:
            return f"Нажата клавиша: {key}"
        case {"type": "scroll", "direction": direction}:
            return f"Прокрутка: {direction}"
        case {"type": event_type}:
            return f"Неизвестное событие: {event_type}"

print(handle_event({"type": "click", "x": 100, "y": 200, "timestamp": 123}))
# Клик в позиции (100, 200)

print(handle_event({"type": "keypress", "key": "Enter"}))
# Нажата клавиша: Enter

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

Шаблоны классов

Одна из мощнейших возможностей — сопоставление с экземплярами классов. Python проверяет тип объекта и может деструктурировать его атрибуты.

from dataclasses import dataclass

@dataclass
class Point:
    x: float
    y: float

@dataclass
class Circle:
    center: Point
    radius: float

@dataclass
class Rectangle:
    top_left: Point
    width: float
    height: float

# Сопоставление с классами
def describe_shape(shape):
    match shape:
        case Circle(center=Point(x=0, y=0), radius=r):
            return f"Круг в начале координат с радиусом {r}"
        case Circle(center=center, radius=r):
            return f"Круг в ({center.x}, {center.y}) с радиусом {r}"
        case Rectangle(width=w, height=h) if w == h:
            return f"Квадрат со стороной {w}"
        case Rectangle(width=w, height=h):
            return f"Прямоугольник {w}x{h}"

print(describe_shape(Circle(Point(0, 0), 5)))
# Круг в начале координат с радиусом 5

print(describe_shape(Circle(Point(3, 4), 2)))
# Круг в (3, 4) с радиусом 2

print(describe_shape(Rectangle(Point(0, 0), 10, 10)))
# Квадрат со стороной 10

Вложенные паттерны

Паттерны можно комбинировать и вкладывать друг в друга для обработки сложных вложенных структур.

# Вложенные паттерны для обработки JSON-подобных данных
def process_response(data):
    match data:
        case {"status": "ok", "data": {"users": [first, *rest]}}:
            return f"Первый пользователь: {first}, ещё {len(rest)}"
        case {"status": "ok", "data": {"users": []}}:
            return "Пользователи не найдены"
        case {"status": "error", "message": msg}:
            return f"Ошибка: {msg}"
        case _:
            return "Неизвестный формат ответа"

print(process_response({
    "status": "ok",
    "data": {"users": ["Алиса", "Боб", "Чарли"]}
}))
# Первый пользователь: Алиса, ещё 2

print(process_response({
    "status": "error",
    "message": "Доступ запрещён"
}))
# Ошибка: Доступ запрещён

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

  • Забытый guard при пересекающихся шаблонах — если шаблон с переменной стоит раньше более конкретного, он перехватит все значения. Всегда размещайте более конкретные шаблоны первыми
  • Мутабельные значения в шаблонах словарей — в шаблоне словаря нельзя использовать вычисляемые ключи, только строковые литералы
  • Неправильное количество элементов — шаблон (x, y) совпадёт только с последовательностью из ровно двух элементов. Для переменного числа используйте *rest

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

Можно ли использовать регулярные выражения в шаблонах? Напрямую нет, но можно использовать guard-условие с re.match(). Например: case s if re.match(r'^\d+$', s):.

Работают ли шаблоны классов с обычными классами (не dataclass)? Да, но для обычных классов нужно явно указывать атрибуты через именованные параметры: case MyClass(attr=value). С dataclass это работает автоматически благодаря __match_args__.

Можно ли комбинировать OR-шаблоны с деструктуризацией? Да: case (0, y) | (y, 0): — совпадёт с точкой на любой из осей. Однако все ветви OR-шаблона должны привязывать одинаковые имена переменных.

Заключение

Продвинутые паттерны match/case в Python — это мощный инструмент для обработки сложных структур данных. Деструктуризация кортежей, списков и словарей, guard-условия, шаблоны классов и вложенные паттерны позволяют писать выразительный и читаемый код для разбора данных, обработки событий и маршрутизации запросов.

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

Стрелочка влевоОператор match/case в Python 3.10+ — основы структурного сопоставленияПрактические примеры match/case в 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 ₽
Подробнее

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