Композиция и агрегация
Тема дорожной карты · Python Programming
Композиция — отношение «has-a»: у Order есть LineItem-ы. Агрегация — композиция с разделяемым внешним владельцем. Обе уменьшают связанность по сравнению с наследованием и упрощают тесты: вместо переопределения методов ты подменяешь сотрудников. Для быстрых value-классов используй dataclasses (или attrs/pydantic), а сотрудников передавай через конструктор. Наследование оставляй на случаи, когда подклассы реально разделяют существенное поведение и соблюдают принцип Лисков; иначе выбирай композицию или протоколы (typing.Protocol).
Как это работает
Композиция и агрегация в Python использует class для связи состояния (атрибуты) и поведения (методы). __init__ инициализирует экземпляры; первый параметр self — сам экземпляр. Методы бывают instance (self), class (@classmethod, cls) или static (@staticmethod). Наследование: class Sub(Base): — допускается множественное; Method Resolution Order (MRO) считается C3-линеаризацией. Dunder-методы (__add__, __len__, __iter__) встраиваются в синтаксис Python (перегрузка операторов, len(), for). @property даёт attribute-like доступ с логикой геттера.
Когда применять
Класс — когда (а) состояние и поведение принадлежат вместе, (б) есть несколько связанных операций над теми же данными, (в) функция с кучей параметров была бы яснее как self-contained объект. Не создавайте класс, если хватит функции или dataclass — Python это не Java; модули с набором функций часто бьют поверхностные иерархии классов. Композиция важнее наследования, кроме случаев настоящего "is-a".
Типичные ошибки
Ловушки Композиция и агрегация: mutable атрибуты на уровне класса, общие между экземплярами (class C: items = [] — каждый C() делит этот список); глубокие иерархии наследования, скрывающие, где живёт поведение (используйте композицию или миксины, максимум 1-2 уровня); перегрузка __eq__ без __hash__ (объект становится unhashable, ломает ключи словаря); злоупотребление @staticmethod для того, что должно быть функцией модуля. __init__ короткий — тяжёлая работа в factory или classmethod.