简单的python继承
class Animal(object):
def __init__(self, nlegs=4):
print '__init__ Animal'
self.nlegs = nlegs
class Cat(Animal):
def __init__(self, talk='meow'):
print '__init__ Cat'
self.talk = talk
class Dog(Animal):
def __init__(self, talk='woof'):
print '__init__ Dog'
self.talk = talk
- 为什么我的猫
tom = Cat()
没有nlegs属性? - 我们应该
Animal.__init__()
从内部显式调用Cat.__init__
,还是应该做更有趣的事情super
? - 如果我想创建一条5条腿的猫该怎么办,是否需要向该
Cat.__init__
接口添加其他参数?
-
要建立在其他人都说过的基础上,是的,您需要调用父
__init__
方法。通常最好使用super。但是,在某些情况下(尤其是当您从多个类继承时),这可能是一个大难题。我将避免赘述,不乏讨论它的各种文章。(此外,其他一些“特殊”功能也有一些奇怪之处。例如,您可以做到,
super(SomeCls, self).__getitem__(5)
但super(SomeCls, self)[5]
不起作用。)至于为什么它的使用是一个好主意一个简单的例子,你可以把
Dog
和Cat
继承Mammal
(从继承Animal
),并没有改变地方比其他类的代码Dog
和Cat
继承。至于您的
tom
实例为什么没有tom.nlegs
的原因,是因为您没有调用Animal
的__init__
方法。还要记住,并不是在初始化时就需要设置所有内容。在这个例子中,它更有意义并不像一套东西
nlegs
的__init__
方法。而是直接在类中进行设置。例如class Mammal(object): nlimbs = 4 def __init__(self): print "I'm a mammal!" class Cat(Mammal): def __init__(self, color="calico"): self.color = color super(Cat, self).__init__() print "I have {0} legs I am {1}".format(self.nlimbs, self.color) class FiveLeggedCat(Cat): nlimbs = 5
基本上,如果某事可能因实例而异(例如,猫的颜色)或需要在初始化时完成(例如,打开文件),则应将其设置为
__init__
。否则,如果我们希望该类的任何实例都相同,那么直接在类定义中进行设置会更干净。
同样,以这种方式设置的属性将可用于文档工具(例如,内置
help
函数),而在初始化时设置的属性将不可用。