Ответ 1
Определенно нехорошо наследовать ребенка от родителя или родителя от ребенка.
Правильный способ сделать это - создать базовый класс, скажем, Person, и наследовать от него и Child, и Parent. Преимущество этого состоит в том, чтобы удалить повторение кода, на данный момент у вас есть только поля имени и фамилии, скопированные в оба объекта, но у вас может быть больше данных или дополнительных методов, таких как get_name()
для работы с этими данными.
Вот пример:
class Person(object):
def __init__(self, firstname, lastname):
self.firstname = firstname
self.lastname = lastname
def get_name(self):
return '%s %s' % (self.firstname, self.lastname)
class Parent(Person):
def __init__(self, firstname, lastname):
super(Parent, self).__init__(firstname, lastname)
self.kids = []
def havechild(self, firstname):
print self.firstname, "is having a child"
self.kids.append(Child(self, firstname))
class Child(Person):
def __init__(self, parent, firstname):
super(Child, self).__init__(firstname, parent.lastname)
self.parent = parent
Другой способ сделать это - сделать это без наследования, но иметь только один объект Person (против Parent и Child). Функция отслеживания статуса семьи и родителей/детей может быть перенесена в другой объект.
Преимущество этого подхода заключается в том, что вы следуете принципу единой ответственности и делаете объекты простыми, каждый объект выполняет только одно.
Вот пример:
from collections import defaultdict
class Person(object):
def __init__(self, firstname, lastname):
self.firstname = firstname
self.lastname = lastname
def get_name(self):
return '%s %s' % (self.firstname, self.lastname)
class FamilyRegistry(object):
def __init__(self):
self.kids = defaultdict(list)
def register_birth(self, parent, child_name):
print parent.firstname, "is having a child"
child = Person(child_name, parent.lastname)
self.kids[parent.lastname].append(child)
return child
def print_children(self, person):
children = self.kids[person.lastname]
if len(children) == 0:
print '%s has no children' % person.get_name()
return
for child in children:
print child.get_name()
Это работает так:
joe = Person('Joe', 'Black')
jill = Person('Jill', 'White')
registry = FamilyRegistry()
registry.register_birth(joe, 'Joe Junior') # Joe is having a child
registry.register_birth(joe, 'Tina') # Joe is having a child
registry.print_children(joe) # Joe Junior Black
# Tina Black
registry.print_children(jill) # Jill White has no children