python虽然是面向对象的语言,但是比起Java来说,在三大特性上没有那么严谨。
继承:python的继承中,不能自动调用父类的构造方法。因此,在子类进行初始化时,必须显式调用父类的构造方法,才能在子类里创建父类的实例属性。
封装:python实质上没有封装,在前面加上(双下划线),也不能实现“封装”。在类的外部依然可以使用“ 类名属性” 、 “对象.类名属性”进行调用;子类依然保留父类的私有属性,也依然可以通过 “子类对象.父类名_父类私有属性”名调用。
多态:python中没有对变量的类型进行规定(只是根据右侧对象的类型规定变量的类型),因此,多态中没有“父类引用指向子类对象”。此外,只要是符合重写方法格式的对象都可以采用这种多态形式,和是否存在继承关系无关。
1.python继承
父类构造方法必须显式调用
子类继承父类除了构造方法以外的全部属性和方法("私有属性"也可以以父类名_私有属性名继承)。因此,子类如果想继承父类的实例属性,必须在自己的构造方法中显式地调用父类的构造方法,调用父类的构造方法有两种格式:
①父类名.__init__(self,父类实例属性)
②super().__init__(父类实例属性)
2.python的多态
只要是符合重写方法格式的对象都可以采用这种多态形式
多态在python里需要满足两个条件:继承,方法重写,和Java不同的是,Java还多一个父类引用指向子类对象,因此下例中的问题就是因为python中缺少这一条件导致的。
class Animal:
def shout(self):
print("动物叫了一声")
class Dog(Animal):
def shout(self):
print("小狗,汪汪汪")
class Cat(Animal):
def shout(self):
print("小猫,喵喵喵")
class Duck: #没有继承Animal类
def shout(self):
print("小鸭,嘎嘎嘎")
class Lover:
def talk(self,animal):
animal.shout()
if __name__ == '__main__':
p1 = Lover()
dog1 = Dog()
cat1 = Cat()
duck1 = Duck()
p1.talk(dog1)
p1.talk(cat1)
p1.talk(duck1)
>>>小狗,汪汪汪
>>>小猫,喵喵喵
>>>小鸭,嘎嘎嘎
上面的代码中,小猫类 和 小狗类都是 Animal类的子类,在Lover类里实现了多态,根据子类对象的不同,调用不同的方法。
但是 鸭子类不是 Animal类的子类,依然可以通过创建Lover类的对象,传入一个本来是用于“多态”的方法talk,依然可以返回自己类里定义的shout()方法。因此,python的多态,不是一种严谨的多态。