Python基础语法——面向对象(下)

2019-07-12  本文已影响0人  竹林徒儿

面向对象三大思想:封装、继承、多态。

可参考一些优质文章:

关于面向对象的相关理解解释:

面向对象基础

装饰器、类之间的关系、继承和多态的理解
面向对象基础进阶


烂笔头:

继承

编写类时,并非总是要从空白开始,如果要编写的类是另一个现次类的特殊版本,可使用继承,一个类继承另一个类时,它将自动获得另一个类的所有属性和方法;原有的类称为父类,而新类称为子类。子类继承了其父类的所有属性和方法,同时还可以定义自己的属性和方法。

子类的方法__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实例。

上一篇 下一篇

猜你喜欢

热点阅读