Python面向对象-类的继承

继承和多态

在面向对象程序设计中,当我们定义一个class的时候,可以从某个现有的class继承,新的class称为子类(Subclass)或派生类,而被继承的class称为基类(Base class)、父类或超类(Super class)。

比如我们已经编写了一个名为Animal的class,有一个run()方法:

class Animal:
    def run(self):
        print('Animal is runing...')

当我们需要编写DogCat类时,就可以直接从Animal类继承:

class Dog(Animal):
    pass


class Cat(Animal):
    pass

对于DogCat来说,Animal就是它们的父类。对于Animal来说,DogCat就是它的子类。

继承有什么好处?最大的好处是子类获得了父类的全部功能。由于Animial实现了run()方法,因此,DogCat作为它的子类,什么事也没干,就自动拥有了run()方法:

dog = Dog()
dog.run()

cat = Cat()
cat.run()

image-20211011152732666

当然,也可以对子类增加一些方法,比如Dog类:

calss Dog:
    def eat(self):
        print('the dog is eating')

image-20211011153152135

继承的第二个好处需要我们对代码做一点改进。你看到了,无论是Dog还是Cat,它们run()的时候,显示的都是Animal is running...,符合逻辑的做法是分别显示Dog is running...Cat is running...,因此,对DogCat类改进如下:

class Dog(Animal):

    def run(self):
        print('Dog is running...')

class Cat(Animal):

    def run(self):
        print('Cat is running...')

image-20211011153427682

当子类和父类都存在相同的run()方法时,我们说,子类的run()覆盖了父类的run(),在代码运行的时候,总是会调用子类的run()。这样,我们就获得了继承的另一个好处:多态。

要理解什么是多态,我们首先要对数据类型再作一点说明。当我们定义一个class的时候,我们实际上就定义了一种数据类型。我们定义的数据类型和Python自带的数据类型,比如str、list、dict没什么两样:

a = list() # a是list类型
b = Animal() # b是Animal类型
c = Dog() # c是Dog类型

判断一个变量是否是某个类型可以用isinstance()判断:

image-20211011153826038

由于DogAnimal的子类,所以创建一个Dog的实例c时,我们认为c的数据类型是Dog,同时c也是Animal

所以:

image-20211011154359129

所以,在继承关系中,如果一个实例的数据类型是某个子类,那它的数据类型也可以被看做是父类。但是,反过来就不行:

image-20211011154530634

子类继承父类构造函数

如果重写了_init_ 时,实例化子类,就不会调用父类已经定义的 _init_,语法格式如下:

class Father(object):
    def __init__(self, name):
        self.name = name
        print("name: %s" % self.name)

    def getName(self):
        return 'Father ' + self.name


class Son(Father):
    def __init__(self, name):
        print("hi")
        self.name = name

    def getName(self):
        return 'Son ' + self.name

son = Son('Python')
print(son.getName())

image-20211011164805662

如果在子类中需要父类的构造方法就需要显式地调用父类的构造方法,或者不重写父类的构造方法。

1.子类不重写 _init_,实例化子类时,会自动调用父类定义的 _init_

class Father(object):
    def __init__(self, name):
        self.name=name
        print ( "name: %s" % self.name )
    def getName(self):
        return 'Father ' + self.name
 
class Son(Father):
    def getName(self):
        return 'Son '+self.name
 
son=Son('runoob')
print ( son.getName() )

image-20211011164905417

如果重写了 _init_ 时,要继承父类的构造方法,需要使用 super 关键字:

# 第一种方法
super(子类, self).__init__(参数1, 参数2...)
# 第二种方法
父类.__init__(self, 参数1, 参数2...)
class Father(object):
    def __init__(self, name):
        self.name = name
        print("name: %s" % self.name)

    def getName(self):
        return 'Father ' + self.name


class Son(Father):
    def __init__(self, name):
        super(Son, self).__init__(name)
        # Father.__init__(self, name)
        print("hi")
        self.name = name

    def getName(self):
        return 'Son ' + self.name

son = Son('Python')
print(son.getName())

image-20211011164645469

posted @ 2021-10-12 14:02  雨-铃  阅读(61)  评论(0编辑  收藏  举报
Tips
复制成功!