Множественное наследование в Python — примеры и MRO

19 июня 2026
Автор

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

Введение

Множественное наследование — это возможность класса наследовать атрибуты и методы сразу от нескольких родительских классов. Python — один из немногих популярных языков, который полноценно поддерживает эту концепцию. В этой статье мы разберём синтаксис, порядок разрешения методов (MRO), паттерн миксинов и типичные подводные камни.

Базовый синтаксис

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

class Flyable:
    def fly(self):
        return "Я умею летать!"


class Swimmable:
    def swim(self):
        return "Я умею плавать!"


class Duck(Flyable, Swimmable):
    def quack(self):
        return "Кря-кря!"


# Утка наследует методы обоих родителей
duck = Duck()
print(duck.fly())    # Я умею летать!
print(duck.swim())   # Я умею плавать!
print(duck.quack())  # Кря-кря!

Класс Duck получает все методы от Flyable и Swimmable, а также определяет свой собственный метод quack.

Порядок разрешения методов (MRO)

Когда несколько родительских классов содержат метод с одинаковым именем, Python использует алгоритм C3-линеаризации для определения порядка поиска. Этот порядок называется MRO (Method Resolution Order):

class A:
    def greet(self):
        return "Привет от A"


class B(A):
    def greet(self):
        return "Привет от B"


class C(A):
    def greet(self):
        return "Привет от C"


class D(B, C):
    pass


d = D()
print(d.greet())  # Привет от B

# Смотрим порядок разрешения методов
print(D.__mro__)
# (<class 'D'>, <class 'B'>, <class 'C'>, <class 'A'>, <class 'object'>)

Python ищет метод в следующем порядке: сначала в самом классе D, затем в B (первый родитель), затем в C, затем в A и наконец в object. Первый найденный метод используется.

Проблема ромбовидного наследования

Ромбовидное наследование (diamond problem) возникает, когда два родительских класса наследуются от одного и того же предка:

class Animal:
    def __init__(self):
        print("Animal.__init__")
        self.type = "животное"


class Flyable(Animal):
    def __init__(self):
        print("Flyable.__init__")
        super().__init__()
        self.can_fly = True


class Swimmable(Animal):
    def __init__(self):
        print("Swimmable.__init__")
        super().__init__()
        self.can_swim = True


class Duck(Flyable, Swimmable):
    def __init__(self):
        print("Duck.__init__")
        super().__init__()


duck = Duck()
# Duck.__init__
# Flyable.__init__
# Swimmable.__init__
# Animal.__init__   <- вызван только ОДИН раз!

print(duck.can_fly)   # True
print(duck.can_swim)  # True
print(duck.type)      # животное

Благодаря super() и MRO, конструктор Animal.__init__ вызывается только один раз, несмотря на то что от Animal наследуются два класса. Это ключевое преимущество Python перед языками, где ромбовидное наследование порождает дублирование.

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

Паттерн миксинов

Миксины — это небольшие классы, которые добавляют конкретную функциональность и не предназначены для самостоятельного использования:

class JsonMixin:
    """Миксин для сериализации объекта в JSON"""
    def to_json(self):
        import json
        return json.dumps(self.__dict__, ensure_ascii=False)


class LogMixin:
    """Миксин для логирования действий"""
    def log(self, message):
        print(f"[LOG] {self.__class__.__name__}: {message}")


class User(JsonMixin, LogMixin):
    def __init__(self, name, email):
        self.name = name
        self.email = email

    def save(self):
        self.log(f"Сохраняем пользователя {self.name}")
        data = self.to_json()
        self.log(f"Данные: {data}")
        return data


user = User("Иван", "ivan@example.com")
user.save()
# [LOG] User: Сохраняем пользователя Иван
# [LOG] User: Данные: {"name": "Иван", "email": "ivan@example.com"}

Миксины — рекомендуемый способ использования множественного наследования. Они делают код модульным и легко тестируемым.

Использование super() при множественном наследовании

При множественном наследовании super() следует MRO, а не указывает на конкретного родителя:

class A:
    def method(self):
        print("A.method")


class B(A):
    def method(self):
        print("B.method")
        super().method()  # Вызовет C.method, а не A.method!


class C(A):
    def method(self):
        print("C.method")
        super().method()  # Вызовет A.method


class D(B, C):
    def method(self):
        print("D.method")
        super().method()  # Вызовет B.method


d = D()
d.method()
# D.method
# B.method
# C.method
# A.method

Это важно понимать: super() в классе B вызывает метод C, а не A, потому что в MRO класса D после B идёт C.

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

  • Конфликт имён атрибутов — когда два родительских класса определяют атрибут с одинаковым именем, используется атрибут из класса, который стоит первым в MRO. Это может привести к неожиданному поведению.
  • Забытый super() в цепочке — если хотя бы один класс в цепочке наследования не вызывает super(), часть конструкторов не будет выполнена.
  • Слишком глубокая иерархия — множественное наследование от 3 и более классов с глубокой иерархией усложняет отладку. Предпочитайте плоские миксины.

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

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

Как узнать MRO класса? Используйте атрибут __mro__ или метод mro(): MyClass.__mro__ или MyClass.mro(). Также можно вызвать help(MyClass) — MRO будет показан в начале вывода.

Чем миксины отличаются от обычного наследования? Миксины — это соглашение, а не отдельный механизм языка. Миксин обычно не имеет своего __init__, не хранит состояние и добавляет только методы. По соглашению, имя миксина заканчивается на Mixin.

Заключение

Множественное наследование в Python — мощный инструмент, который при правильном использовании упрощает код. Ключ к успеху — понимание MRO, правильное использование super() и применение паттерна миксинов вместо сложных иерархий.

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

Стрелочка влевоНаследование классов в Python — основы и примерыКонтекстный менеджер with в 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 ₽
Подробнее

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