Проблема наследования класса Python
Я играю с наследованием класса Python и сталкивается с проблемой, когда унаследованный __init__
не выполняется, если вызван из подкласса (код ниже), результат, который я получаю из Active Python, это:
>>> start
Tom Sneed
Sue Ann
Traceback (most recent call last):
File "C:\Python26\Lib\site-packages\pythonwin\pywin\framework\scriptutils.py", line 312, <br>in RunScript
exec codeObject in __main__.__dict__
File "C:\temp\classtest.py", line 22, in <module>
print y.get_emp()
File "C:\temp\classtest.py", line 16, in get_emp
return self.FirstName + ' ' + 'abc'
AttributeError: Employee instance has no attribute 'FirstName'
Здесь код
class Person():
AnotherName = 'Sue Ann'
def __init__(self):
self.FirstName = 'Tom'
self.LastName = 'Sneed'
def get_name(self):
return self.FirstName + ' ' + self.LastName
class Employee(Person):
def __init__(self):
self.empnum = 'abc123'
def get_emp(self):
print self.AnotherName
return self.FirstName + ' ' + 'abc'
x = Person()
y = Employee()
print 'start'
print x.get_name()
print y.get_emp()
Ответы
Ответ 1
Три вещи:
- Вам нужно явно вызвать конструктор. Он не вызывается для вас автоматически, как в С++
- Используйте класс нового стиля, унаследованный от объекта
- С классом нового стиля используйте доступный метод super()
Это будет выглядеть так:
class Person(object):
AnotherName = 'Sue Ann'
def __init__(self):
super(Person, self).__init__()
self.FirstName = 'Tom'
self.LastName = 'Sneed'
def get_name(self):
return self.FirstName + ' ' + self.LastName
class Employee(Person):
def __init__(self):
super(Employee, self).__init__()
self.empnum = 'abc123'
def get_emp(self):
print self.AnotherName
return self.FirstName + ' ' + 'abc'
Рекомендуется использовать супер, так как он также будет правильно обрабатывать конструкторы вызовов только один раз в нескольких случаях наследования (пока каждый класс в графе наследования также использует супер). Это также одно меньшее место, где вам нужно изменить код, если/когда вы меняете то, на что наследуется класс (например, вы определяете базовый класс и изменяете вывод и не должны беспокоиться о своих классах, вызывающих неверного родителя конструкторы). Также на фронте MI вам нужен только один супервызов для правильного вызова всех конструкторов базового класса.
Ответ 2
Вы должны явно вызвать функцию init для суперкласса:
class Employee(Person):
def __init__(self):
Person.__init__(self)
self.empnum = "abc123"
Ответ 3
Сотрудник должен явно вызвать родителя __init__ (не init):
class Employee(Person):
def __init__(self):
Person.__init__(self)
self.empnum = 'abc123'
Ответ 4
Вместо шаблона super(class, instance)
почему бы просто не использовать super(instance)
, поскольку класс всегда instance.__class__
?
Существуют ли конкретные случаи, когда это не было бы instance.__class__
?