Как правильно переопределить конструктор класса? Приветствую. Да, вопрос похож на нубство, но я как-то совсем запутался. Есть класс, унаследованный от datetime.date. К этому классу нужно добавить пару атрибутов и пару свойств. Для этого я в своём классе прописывают функцию __init__, в которой делаю обычную рутину: проверяю параметры, создаю новые атрибуты, после чего вызываю конструктор класса-родителя. Когда я пытаюсь создать экземпляр класса, используя только те атрибуты, которые принимает родительский класс, всё отлично. Когда же я пытаюсь добавить ещё атрибутов, вылетает TypeError. ЧЯДНТ? Немного кода. Так выглядит мой класс.class MyClass(date): def __init__(self, m_year: int, m_month: int, m_day: int, expiry: date=None, *args, **kwargs): if expiry: self._expiry = expiry else: self._expiry = date(m_year, m_month, m_day) super(Maturity, self).__init__(*args, **kwargs) @property def expiry(self): return {'year': self._expiry.year, 'month': self._expiry.month, 'day': self._expiry.day} Создаём экземпляр класса. Так всё хорошо, работают как методы родительского класса, так и моего.>>> q = MyClass(2015, 5, 5) >>> q MyClass(2015, 5, 5) >>> q.strftime('%Y') '2015' >>> q.expiry {'day': 5, 'month': 5, 'year': 2015} Но стоит мне захотеть болшего, как выпадает вот такое исключение.>>> q = MyClass(2015, 5, 5, expiry=date(2015, 5, 1)) TypeError: function takes at most 3 arguments (4 given) Помимо прочего, смущает то, что код одинаково работает как с 8-й строкой (вызов super), так и без неё.
Проблема здесь в том, что при вызове конструктора родительского класса, вы пытаетесь передать все аргументы, включая те, которые предназначены только для вашего класса. Для того чтобы передать только те аргументы, которые родительский класс может принять, вам нужно явно указать их при вызове super().init().
Проблема здесь в том, что при вызове конструктора родительского класса, вы пытаетесь передать все аргументы, включая те, которые предназначены только для вашего класса. Для того чтобы передать только те аргументы, которые родительский класс может принять, вам нужно явно указать их при вызове super().init().
Используйте следующий код для вашего класса:
class MyClass(date):def __init__(self, m_year: int, m_month: int, m_day: int, expiry: date=None, *args, **kwargs):
if expiry:
self._expiry = expiry
else:
self._expiry = date(m_year, m_month, m_day)
super(MyClass, self).__init__(m_year, m_month, m_day, *args, **kwargs)
@property
def expiry(self):
return {'year': self._expiry.year, 'month': self._expiry.month, 'day': self._expiry.day}
Теперь при создании экземпляра класса с указанием дополнительного атрибута expiry, код должен успешно выполниться без ошибок:
q = MyClass(2015, 5, 5, expiry=date(2015, 5, 1))print(q)
print(q.strftime('%Y'))
print(q.expiry)
Попробуйте это исправление и проверьте, работает ли код корректно. Если у вас остались дополнительные вопросы, не стесняйтесь задать их.