Python – 面向对象编程 – 三大特性之继承

继承

  • 继承也是面向对象编程三大特性之一
  • 继承是类与类的一种关系
  • 定义一个新的 class 时,可以从某个现有的 class 继承
  • 新的 class 类就叫子类(Subclass)
  • 被继承的类一般称为父类、基类、超类
  • 通俗理解:子类与父类的关系,即爸爸与儿子,爸爸有了个儿子,儿子继承爸爸的属性和方法,爸爸的爸爸就是爷爷…以此类推

 

继承的实际栗子

  • 猫、狗都是动物,所以动物是父类,猫、狗是动物的子类
  • 小菠萝、大菠萝都是人类,所以人类是父类,小菠萝、大菠萝是人类的子类
  • 动物、人类都是生物,所以生物是父类,动物、人类是生物的子类
  • 那么一般称生物类是猫、狗、小菠萝、大菠萝的超类、祖父类

 

继承的好处

  • 实现代码的重用,相同的代码不需要重复编写
  • 子类拥有父类的所有属性、方法,提高了代码的可扩展性和重用性
  • 在子类继承父类的同时,也可以重新定义某些属性,并重写某些方法,即覆盖父类的属性、方法,变成子类独有的属性、方法
  • 子类也可以添加新的属性、方法

 

继承的使用场景

  • 假如我需要定义几个类,而类与类之间有一些公共的属性和方法
  • 可以把公共的属性和方法提取作为父类(基类)的属性、方法,而特殊的属性、方法则在本类中定义
  • 比如猫、狗都有吃喝拉撒四种行为,这些就是公共方法;但只有猫会喵喵叫,只有狗会汪汪叫,这些就是特殊方法

具体可以看下面的【不使用继承、使用继承的区别】

 

继承有几种?

  1. 单继承
  2. 多继承

 

Python 中继承的简单语法

class Parent:
    pass

class Child(Parent):
    pass

  

有个重点

所有类的都有一个共同父类,就是 object,默认会继承它,写或不写都一样

class Parent:
    pass

class Parent(object):
    pass

 

不使用继承、使用继承的区别

需求背景

有一个动物类、狗类、猫类

  • 动物类有吃、喝、跑方法
  • 狗类有吃、喝、跑、汪汪叫方法
  • 猫类有吃、喝、跑、喵喵叫方法

 

不使用继承

类图

 

需求分析

  • 每个类都拥有自己的方法
  • 他们都拥有吃、喝、拉方法,且功能一样

 

实际代码

# 不使用继承
class Animal:
    def eat(self):
        print("吃东西...")

    def drink(self):
        print("喝水...")

    def run(self):
        print("跑步...")


class Cat:
    def eat(self):
        print("吃东西...")

    def drink(self):
        print("喝水...")

    def run(self):
        print("跑步...")

    def miao(self):
        print("喵喵叫...")


class Dog:
    def eat(self):
        print("吃东西...")

    def drink(self):
        print("喝水...")

    def run(self):
        print("跑步...")

    def wang(self):
        print("汪汪叫...")

 

这样写有什么问题?

  • 三个类,需要重复写三次吃、喝、跑的方法
  • 假设此时需要修改 吃 方法的代码,那么还得同步修改三个类的 吃 方法,属于不必要的重复性工作
  • 使用继承可以大大减少开发工作量

 

使用继承

类图

从类图就能看到,使用继承就变得简单多了

 

需求分析

  • 提取三个类的公共方法,吃、喝、跑,然后抽象成动物类
  • 所以动物类是一个父类,狗类、猫类继承它

 

实际代码

# 使用继承
class Animal:
    def eat(self):
        print("吃东西...")

    def drink(self):
        print("喝水...")

    def run(self):
        print("跑步...")


class Cat(Animal):
    def miao(self):
        print("喵喵叫...")


class Dog(Animal):
    def wang(self):
        print("汪汪叫...")

 

优势

  • 子类继承父类,可以直接享受父类已经封装好的方法,不再需要自己开发一次
  • 子类可以根据自身的需要,封装子类独有的属性、方法,比如猫类的喵喵叫,狗类的汪汪叫 

 

专业术语总结

  • Dog 类是 Animale 类的子类,Animal 类是 Dog 类的父类,Dog 类从 Animal 类继承
  • Dog 类是 Animale 类的派生类,Animal 类是 Dog 类的基类,Dog 类从 Animal 类派生

  

继承的传递性

什么是传递性

通俗来讲

  • C 类从 B 类继承,B 类又从 A 类继承
  • 那么 C 类会拥有 A、B 类的所有属性和方法

 

官方来讲

子类拥有父类以及父类的父类,以及所有父类的父类的父类…中封装的所有属性、方法

 

实际栗子

  • 还是拿上面的动物类、狗类、猫类继续展开讲
  • 假设此时有一个新的类叫柴犬类,继承于狗类,有独有的方法摇尾巴

 

类图

 

实际代码

class Animal:
    def eat(self):
        print("吃东西...")

    def drink(self):
        print("喝水...")

    def run(self):
        print("跑步...")


class Cat(Animal):
    def miao(self):
        print("喵喵叫...")


class Dog(Animal):
    def wang(self):
        print("汪汪叫...")


class Chai(Dog):
    def shake(self):
        print("小柴柴摇尾巴...")


chai = Chai()
# 调用 父类的父类 的方法
chai.eat()
# 调用 父类 的方法
chai.wang()
# 调用 自己独有 的方法
chai.shake()


# 输出结果
吃东西...
汪汪叫...
小柴柴摇尾巴...
  • Chai 类拥有 Dog 类、Animal 类所有属性、方法
  • 但它不会拥有 Cat 类的属性、方法,因为他们没有继承关系

 

继承和抽象

抽象即抽取类似或者说比较像的部分

 

小栗子

  • 从小菠萝、小韵韵两个对象中,抽取公共部分抽象成父类
  • 再从人,猪,狗这三个类,抽取公共部分抽象成父类

 

继承的重点

  • 是基于抽象的结果,通过编程语言去实现它
  • 先经历抽象这个过程,才能通过继承的方式去表达出抽象的结构
  • 抽象只是分析和设计的过程中,一个动作或者说一个技巧,通过抽象得到一个类

  

代码栗子

class Animal:
    def __init__(self, name, age, sex):
        self.__name = name
        self.__age = age
        self.__sex = sex

    # 提取的公共方法
    def eat(self):
        print(f"{self.__name} 正在吃饭....")

    @property
    def name(self):
        return self.__name


class Person(Animal):
    def walk(self):
        print(f"{self.name} 两条腿正在走路")


class Pig(Animal):
    def kill(self):
        print(f"{self.name} 猪正在去屠宰场ing...")


class Dog(Animal):
    def wang(self):
        print(f"{self.name} 汪汪叫ing...")


person = Person("小菠萝", 25, "Male")
pig = Pig("麦兜", 4, "")
dog = Dog("柴犬", 3, "")

# 调用父类的公共方法
person.eat()
pig.eat()
dog.eat()

# 调用子类独有的方法
person.walk()
pig.kill()
dog.wang()



# 输出结果
小菠萝 正在吃饭....
麦兜 正在吃饭....
柴犬 正在吃饭....

小菠萝 两条腿正在走路
麦兜 猪正在去屠宰场ing...
柴犬 汪汪叫ing...