面向对象三大思想:封装、继承、多态。
可参考一些优质文章:
关于面向对象的相关理解解释:
装饰器、类之间的关系、继承和多态的理解
面向对象基础进阶
烂笔头:
继承
编写类时,并非总是要从空白开始,如果要编写的类是另一个现次类的特殊版本,可使用继承,一个类继承另一个类时,它将自动获得另一个类的所有属性和方法;原有的类称为父类,而新类称为子类。子类继承了其父类的所有属性和方法,同时还可以定义自己的属性和方法。
子类的方法__init__()
创建子类的实例时,Python首先需要完成的任务是给父类的所有属性赋值。为此,子类的方法__init__()
需要父类施以援手。
例如,上文中的关于汽车的类
class Car():
"""一次模拟汽车的简单尝试 """
def __init__(self,make,modle,year):
"""初始化描述汽车的属性 """
self.make = make
self.modle = modle
self.year = year
def get_descriptive_name(self):
"""返回整洁的描述性信息 """
long_name = str(self.year) + ' ' + self.make + ' ' self.modle
return long_name.title()
my_new_car = Car('audi','a4',2016)
下面如果我们需要模拟电动汽车。电动汽车是一种特殊的汽车,因此我们可以在前面创建的Car类的基础上创建ElectricCar,这样我们就只需为电动汽车特有的属性和行为编写代码。
下面来创建一个简单的ElectricCar类版本,它具有Car类所有的功能:
class Car():
"""一次模拟汽车的简单尝试 """
def __init__(self,make,modle,year):
"""初始化描述汽车的属性 """
self.make = make
self.modle = modle
self.year = year
def get_descriptive_name(self):
"""返回整洁的描述性信息 """
long_name = str(self.year) + ' ' + self.make + ' ' self.modle
return long_name.title()
class ElectricCar(Car):
"""电动汽车的独特之处"""
def __init__(self,make,modle,year):
"""
初始化父类的属性,再初始化电动汽车的特有属性
"""
super().__init__(make,model,year)
self.battery_size = 70
def describe_battery(self):
"""打印一条电瓶容量的消息"""
print("This car has a " + str(self.battery_size) + "-kwh battery".)
my_tesla = ElectricCar('tesla','model s','2016')
my_tesla.describe_battery()
首先是Car类的代码,创建子类时,父类必须包含再当前文件夹中,且位于子类前面。
定义子类时,必须在括号内指定父类的名称。方法init()接受创建Car实例所需的信息
其中super()是一个特殊函数,帮助Python将父类和子类关联起来。这行代码让Python调用ElectricCar的父类的方法__init()
,让ElectricCar实例包含父类的所有属性。父类也称为超类(superclass),名称super因此而得名。
重写父类的方法
对于父类的方法,只要它不符合子类模拟的实物的行为,都可对其进行 重写。为此可在子类中定义一个这样的方法,即它与要重写的父类方法同名。这样,Python将不会考虑这个父类方法,而只关注你在子类中定义的相应方法。
假设Car类有一个名为fill_gas_tank()的方法,它对全自动汽车来说毫无意义,因此你可能想重写它:
class ElectricCar(Car):
--snip--
def fill_gas_tank():
"""电动汽车没有油箱"""
print("This car doesn't need a gas tank! ")
现在,如果有人对电动汽车调用方法fill_gas_tank(),Python将忽略Car类中的方法fill_gas_tank(),转而运行上述代码。
使用继承时,可以让子类保留从父类那里继承而来的精华而剔除不需要的糟粕
将实例用作属性
class Car():
--snip--
class Battery():
"""一次模拟电动汽车电瓶的简单尝试"""
def __init(self,battery_size=70):
"""初始化电瓶的属性"""
self.battery_size = battery_size
def describe_battery(self):
"""打印一条描述电瓶容量的消息 """
print(“This is a “ str(self.battery_size) + "-kwh battery.")
class ElectricCar(Car):
"""电动汽车的特殊之处"""
def __init__(self,make,modle,year):
#***************************************
self..battery = Battery()
#***************************************
在ElectricCar类中,我们添加了一个名为self.battery的属性。这行代码让Python创建一个新的Battery实例(由于没有指定尺寸,因此默认值70),并将该实例存储在属性self.battery中,每当方法__init__()
被调用时, 都将执行该操作;因此现在每个ElectricCar实例都包含一个自动创建的Battery实例。