Python. Наследование

Перевод параграфа 6.4 Inheritance из книги Intermediate Python.

Наследование — это механизм создания новых классов. Наследники специализируют или изменяют базовые классы добавляя в них новую функциональность. Python поддерживает множественное наследование как C++. Пример одиночного наследования в Python:

Ключевое слово super

В одиночной иерархии наследования super используется для ссылки на родительский класс без явного указания на него. Это похоже на метод super в Java. super применяется при переопределении метода, когда требуется вызвать родительскую версию переопределяемого метода. В примере выше, метод __init__ в классе SavingsAccount переопределён, но с помощью super вызывается метод __init__ родительского класса. При множественном наследовании super играет более важную роль.

Множественное наследование

В множественном наследовании класс может иметь несколько родительских классов. Одной из сложностей такого вида наследования является поиск нужной версии метода при его вызове. Представим что класс D наследник классов B и C и нужно вызывать метод родительского класса когда оба родителя имеют этот метод. В Python эта ситуация решается алгоритмом Method Resolution Order (MRO), он определяет как производится поиск метода в классе и всех базовых классах. Порядок методов вычисляется в момент определения класса и сохраняется в поле класса __mro__.

Алгоритм MRO состоит из двух шагов: поиск в глубину слева направо по всем классам в иерархии и удаление повторявшихся классов, кроме последнего вхождения.

В примере выше поиск в глубину по всем классам выдаёт такой список классов:

Затем удаление повторяющихся классов, кроме последнего вхождения:

Обратите внимание что если предок явно не задан, то класс наследуются от класса object.

Множественном наследование и super

Рассмотрим иерархию из прошлого примера. Класс A объявляет метод переопределяемый классами B, C и D. Предположим есть требование чтобы все методы вызывались, они сохраняют данные в каждом классе где объявлены, пропуск вызова приведёт к потере данных. Такое требование реализуется с помощью метода super и MRO:

Предположим метод self.meth() вызывается из экземпляра класса D. Тогда super(D, self).meth() найдёт и вызовет метод B.meth(self), так как B первый базовый класс следующий за D в D.__mro__ и B определяет метод meth.

Теперь B.meth вызывает super(B, self).meth() и так как self является экземпляром класса D, следующий класс после B это C (D.__mro__ = [D, B, C, A]) и поиск meth продолжается.

Следующим вызывается C.meth, который в свою очередь вызывает super(C, self).meth().

Следующий после C класс в MRO это A и вызывается метод A.meth. Это исходное определение meth, поэтому вызов super не производится.

Используя super и MRO, интерпретатор находит и вызывает все версии метода meth реализованные в каждом класс иерархии.

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

Если вы нашли ошибку, пожалуйста, выделите фрагмент текста и нажмите Ctrl+Enter.

Добавить комментарий

Ваш e-mail не будет опубликован. Обязательные поля помечены *