Проблема наследования классов Python - PullRequest
16 голосов
/ 30 мая 2009

Я играю с наследованием Python Class и столкнулся с проблемой, когда унаследованный __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()

Ответы [ 4 ]

25 голосов
/ 30 мая 2009

Три вещи:

  1. Вам нужно явно вызвать конструктор. Он не вызывается для вас автоматически, как в C ++
  2. Использовать класс нового стиля, унаследованный от объекта
  3. С классом нового стиля используйте доступный метод 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'

Рекомендуется использовать super, так как он также правильно обрабатывает вызовы конструкторов только один раз в нескольких случаях наследования (при условии, что каждый класс в графе наследования также использует super). Это также на одно место меньше, чем нужно менять код, если / когда вы меняете то, от чего унаследован класс (например, вы выделяете базовый класс и изменяете наследование, и вам не нужно беспокоиться о том, что ваши классы вызывают неправильный родитель конструкторы). Также на фронте MI вам нужен только один супер-вызов для правильного вызова всех конструкторов базового класса.

9 голосов
/ 30 мая 2009

Вы должны явно вызвать функцию init суперкласса:

class Employee(Person):
    def __init__(self):
        Person.__init__(self)
        self.empnum = "abc123"
5 голосов
/ 30 мая 2009

Сотрудник должен явно вызвать __init__ родителя (не init):

 class Employee(Person):  
    def __init__(self):  
         Person.__init__(self)  
         self.empnum = 'abc123'  
2 голосов
/ 10 ноября 2010

Вместо шаблона super(class, instance), почему бы просто не использовать super(instance), поскольку класс всегда instance.__class__?

Есть ли конкретные случаи, когда это не будет instance.__class__?

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...