Подтвердить что ты не робот

Проблема наследования класса 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()
4b9b3361

Ответ 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__?