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

继承最大好处是,是子类获得了父类的全部功能

例如我们定义一个类Cars,

class Cars(object):
    def run(self):
        print 'car start runing!'

 写两个类,让他们继承Cars

class Audi(Cars):
    pass


class BMW(Cars):
    pass

 再分别实例化,分别调用run()

audi = Audi()
audi.run()

bmw = BMW()
bmw.run()

可以看到,2个子类什么方法都没有写,但是调用run()后,都有打印结果

car start runing!
car start runing!

 当子类和父类都存在相同的run()方法时,我们说,子类的run()覆盖了父类的run(),在代码运行的时候,总是会调用子类的run()

class Audi(Cars):
    def run(self):
        print 'Audi start runing!'


class BMW(Cars):
    def run(self):
        print 'BMW start runing!'
-------------------------------------
Audi start runing!
BMW start runing!

 

这样,我们就获得了继承的另一个好处:多态。

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

a = list()  # a是list类型
b = Cars()  # b是Cars类型
c = Audi()  # c是Audi类型

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

print isinstance(a, list)
print isinstance(b, Cars)
print isinstance(c, Audi)
print isinstance(c, Cars)
print isinstance(b, Audi)
-----------------------------
True
True
True
True
False

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

要理解多态的好处,我们还需要再编写一个函数,这个函数接受一个Cars类型的变量: 

def run_test(cars):
    cars.run()  

 当传入Cars实例时,run_test()就会打印

run_test(Cars())
-----------------
car start runing!

 当传入Audi和BWM的实例时,run_test()就会打印

run_test(Audi())
run_test(BMW())
--------------------
Audi start runing!
BMW start runing!

如果在增加一个类,继承自Cars类

class Bike(Cars):
    def run(self):
        print 'bike..........'

run_test(Bike())
------------------------
bike..........

 新增一个Cars的子类,不必对run_test()做任何修改,实际上,任何依赖Cars作为参数的函数或者方法都可以不加修改地正常运行,原因就在于多态。

多态的好处就是,当我们需要传入子类时,我们只需要接收Cars类型就可以了,因为Audi,BWM,Bike……都是Cars类型,然后,按照Cars类型进行操作即可。由于Cars类型有run()方法,因此,传入的任意类型,只要是Animal类或者子类,就会自动调用实际类型的run()方法,这就是多态的意思。

对于一个变量,我们只需要知道它是Cars类型,无需确切地知道它的子类型,就可以放心地调用run()方法,而具体调用的run()方法是作用在Cars、Audi,BMWt还是Bike对象上,由运行时该对象的确切类型决定,这就是多态真正的威力:调用方只管调用,不管细节,而当我们新增一种Cars的子类时,只要确保run()方法编写正确,不用管原来的代码是如何调用的。这就是著名的“开闭”原则:

对扩展开放:允许新增Cars子类;

对修改封闭:不需要修改依赖Animal类型的run_test()等函数。

 

继承可以把父类的所有功能都直接拿过来,这样就不必重零做起,子类只需要新增自己特有的方法,也可以把父类不适合的方法覆盖重写;

有了继承,才能有多态。在调用类实例方法的时候,尽量把变量视作父类类型,这样,所有子类类型都可以正常被接收。

 

 posted on 2017-04-19 23:28  残夜天晴  阅读(188)  评论(0编辑  收藏  举报