Python零基础入门篇 - 37 - 类的继承、多态与多重继承
前言:
✌ 作者简介:渴望力量的哈士奇,大家可以叫我 🐶哈士奇🐶 。(我真的养了一只哈士奇)
📫 如果文章知识点有错误的地方,请指正!和大家一起学习,一起进步👀
🔥 如果感觉博主的文章还不错的话,还请不吝👍关注、点赞、收藏三连支持👍一下博主哦
💬 人生格言:优于别人,并不高贵,真正的高贵应该是优于过去的自己。💬
🎉🎉欢迎持续关注🎉🎉 |
文章目录
类的继承
什么是继承?
举个生活中的例子:小明是人类,小明的爸爸也是人类。小明的爸爸会说话、会走路… 同样的小明也会说话、会走路。那么小明打爸爸与小明的这次关系就叫做 继承
;小明的爸爸是小明的父亲(父类),小明是爸爸的儿子(子类);在扩大一个层面来说,小明和小明的爸爸都是人类,所以人类就是小明和小明爸爸的父类,而小明和小明爸爸就是人类的子类。(我也不明白我为啥要扯这种废话…),结合这种关系,我们再来看看代码中 类 的继承关系
-
与前文我们在初识面向对象编程中提到的封装一样,继承也是面向对象编程三大特性之一
-
在编程中继承的关系是类与类的一种关系
-
当我们定义了一个新的 类 时,可以从当前存在的 类 通过继承的关系得到其功能、属性等;被继承的 类 我们就叫做父类(也可以叫做基类或者超类),而新的 类 则叫做子类,且其具备父类的功能、属性与一些特性。
父类与子类
- 首先,子类继承了父类,那么子类就拥有了父类的所有属性、方法。实现了代码的重用,那么相同的代码不需要重复编写,极大的提高了代码的可扩展性和重用性
- 子类在通过继承拥有了父类的所有属性、方法后,也可以编写仅属于自己的新属性、新方法等,与父类并不冲突。父类不具备子类自有的属性与方法
如何使用继承
我们来看一个例子:
class Father(object):# 定义一个 Father 类,有两个函数 talk() 与 run() def talk(self): print('会说话') def run(self): print('会跑步')class Son(Father):# 定义一个 Son 类,小括号内没有传入 object 通用类,传入的是 Father类 # 传入的 Father 类,就是要继承的类;也就是父类,也叫做基类(基础类的简称) # object 为 类 继承的通用类 def swim(self): print('\'Son\' 会游泳')son = Son()print(son.talk(), son.run(), son.swim())# >>> 执行结果如下:# >>> 会说话# >>> 会跑步# >>> 'Son' 会游泳# >>> 思考一个问题:既然 Son 类 集成了 Father 类,那么 Father 类 是否可以调用 Son 类 的函数?father = Father()print(father.swim)# >>> 执行结果如下:# >>> AttributeError: 'Father' object has no attribute 'swim'
结合上面的示例我们得出以下结论:
- 定义子类时,我们需要将父类传入子类的参数内
- 子类实例化之后可以调用 自己 与 父类 的函数与变量
- 父类则无法调用子类的独有函数与变量
关于继承的小练习
下面我们根据上面的示例,做一个关于继承的小练习,加深我们对继承的理解。
class Father(object): def __init__(self, name, sex): self.name = name self.sex = sex def talk(self): return f'{self.name} 会说话' def is_sex(self): if self.sex == 'man': return f'{self.name} 是 男的。' else: return f'{self.name} 是 女的。'class Son01(Father): def play_football(self): return f'{self.name} 会踢足球,'class Son02(Father): def play_basketball(self): return f'{self.name} 会打篮球,'son01 = Son01('蛋蛋', 'man')result_son01 = son01.play_football()print(result_son01, son01.is_sex())son02 = Son02('花花', 'woman')result_son02 = son02.play_basketball()print(result_son02, son02.is_sex())father = Father('蛋蛋与花花的爸爸', 'man')result = father.is_sex()print(result)# >>> 执行结果如下:# >>> 蛋蛋 会踢足球, 蛋蛋 是 男的。# >>> 花花 会打篮球, 花花 是 女的。# >>> 蛋蛋与花花的爸爸 是 男的。
拓展:继承的传递性
什么是传递性?
关于 继承的传递性
,官方的解释为:子类拥有父类以及父类的父类,以及所有父类的父类的父类…中封装的所有属性、方法。
通俗的来说就是 A类 被 B类 继承, B类 又被 C类 继承,那么 C类 就会拥有 A、B 类的所有属性和方法。
举一个生活中的真实场景:动物是一个大类,在动物大类之下有猫、狗等不同分类,猫又有橘猫、狸花猫、布偶猫等品种。狗也一样,有哈士奇、萨摩耶、吉娃娃等犬种。
现在我们定义一个 哈士奇 继承于 狗类、狗类继承于动物类,且哈士奇拥有拉雪橇的方法。
代码示例如下:
class Animal(object): def __init__(self, name): self.name = name def eat(self): print(f"{self.name} 会吃东西...") def drink(self): print(f"{self.name} 会喝水...")class Cat(Animal): def miao(self): print(f"{self.name} 会喵喵叫...")class Dog(Animal): def wang(self): print(f"{self.name} 会汪汪叫...")class Husky(Dog): def work(self): print(f"{self.name} 会拉雪橇...")husky = Husky('哈士奇')husky.eat() # 调用 父类 的 父类 的 方法husky.wang() # 调用 父类 的方法husky.work() # 调用 自己独有 的方法# print(husky.eat(), husky.wang(), husky.work())# >>> 执行结果如下:# >>> 哈士奇 会吃东西...# >>> 哈士奇 汪汪叫...# >>> 哈士奇 会拉雪橇...
- husky 类拥有 Dog 类、Animal 类所有属性、方法
- 但它不会拥有 Cat 类的属性、方法,因为他们没有继承关系
类的多态
什么是类的多态? —> 虽然拥有相同的功能,但是却表现出了多种的状态。联想到 类 ,虽然通过 继承 拥有了相同的函数,但是执行的结果却不尽相同。这就是类的多态。
当子类继承了父类的函数,如何才能让子类的父类函数拥有自己的状态呢?答案很简单,那就是在子类中重写父类的方法。
代码示例如下:
# 创建一个父类class XiaoMing_Father(object): def talk(self): print('小明的爸爸发表了一个观点。')# 创建一个子类继承了父类class XiaoMing_Brother(XiaoMing_Father): def run(self): print('小明的哥哥不同意爸爸的观点,自己跑了。')if __name__ == '__main__': xiaoming_brother = XiaoMing_Brother() xiaoming_brother.talk() # >>> 执行结果如下:# >>> 小明的爸爸发表了一个观点。# >>> xiaoming_brother 实例化之后继承了 XiaoMing_Father 类 的 talk() 函数# >>> 但是我们并不想要 XiaoMing_Father 类 的 talk() 函数的输出结果# >>> 这个时候就用到了 '多态',我们可以在 XiaoMing_Brother 类 中重写 talk() 函数# >>> 这样就可以让 XiaoMing_Brother 类 虽然继承了 XiaoMing_Father 类 的 talk() 函数# >>> 但是我们就拥有了自己的状态。示例如下:
# 创建一个父类class XiaoMing_Father(object): def talk(self): print('小明的爸爸发表了一个观点。')# 创建一个子类继承了父类class XiaoMing_Brother(XiaoMing_Father): def run(self): print('小明的哥哥不同意爸爸的观点,自己跑了。') def talk(self): print('小明的哥哥不同意爸爸的观点,并发表了自己的不同观点') # 再创建一个子类class XiaoMing(XiaoMing_Father): def talk(self): print('小明的哥哥也不同意爸爸的观点,但也不支持哥哥的看法') if __name__ == '__main__': xiaoming_father = XiaoMing_Father() xiaoming_father.talk() xiaoming_brother = XiaoMing_Brother() xiaoming_brother.talk() xiaoming = XiaoMing() xiaoming.talk() # >>> 执行结果如下:# >>> 小明的爸爸发表了一个观点。# >>> 小明的哥哥不同意爸爸的观点,并发表了自己的不同观点# >>> 小明的哥哥也不同意爸爸的观点,但也不支持哥哥的看法# >>> 这里我们看到 xiaoming_brother、xiaoming 都继承了 XiaoMing_Father 类 的 talk() 函数# >>> 虽然都拥有相同的 talk() 函数,但是他们返回的结果却是不同的。
这里我们思考一下,为什么要有多态,为什么要去继承父类?
其实这是为了使用已经写好的类中的函数,为了保留子类中某个和父类名称一样的函数的功能。这个时候就需要使用到类的多态了。
同时可以帮助我们保留子类中的函数功能。
Python 中的 supper 函数
supper() 函数的作用:在 Python 中 ,子类继承父类的方法而使用的关键字;当子类继承父类之后,就可以完全使用父类的方法了。
supper() 函数的用法:示例如下
class Father(object): def __init__(self): print('Hello,I\'m Father') class Son(Father): def __init__(self): print('Hello,I\'m Son') super(Son, self).__init__() # 使用 supper() 函数 令 Son() 函数执行 Father 类的 构造函数 # Son :当前类 # self :类的实例 # __init__ : 使用父类的方法 # 在 Python 2.x 时代,supper() 函数内的两个参数是必须传入的;Python 3.x 时代之后可不传 if __name__ == '__main__': son = Son() # >>> 执行结果如下:# >>> Hello,I'm Son# >>> Hello,I'm Father# >>> 可以看到除了执行 Son() 类的自身的构造函数之外,同时也执行了 Father() 类的构造函数# >>> 这就是 supper() 函数的使用方法
那么我们能够利用 supper() 函数 做什么呢?其实继承了父类函数之后,也会继承父类构造函数的初始化值,接下来我们来看看 supper() 函数 如何来传值。
class Father(object): def __init__(self, father_name): print('Hello,I\'m Father %s' % father_name)class Son(Father): def __init__(self, son_name, father_name): print('Hello,I\'m Son %s' % son_name) super(Son, self).__init__(father_name) # 使用 supper() 函数 令 Son() 函数执行 Father 类的 构造函数 # Son :当前类 # self :类的实例 # __init__ : 使用父类的方法 # 在 Python 2.x 时代,supper() 函数内的两个参数是必须传入的;Python 3.x 时代之后可不传if __name__ == '__main__': son = Son('Neo', 'Neo\'s_father') # >>> 执行结果如下:# >>> Hello,I'm Son Neo# >>> Hello,I'm Father Neo's_father
上面的演示案例是基于 Python 2.x 时代,supper() 函数内的两个参数是必须传入的方式实现的,我们再看看Python 3.x 时代之后实现的方式
class Father(object): def __init__(self, father_name): print('Hello,I\'m Father %s' % father_name)class Son(Father): def __init__(self, son_name): print('Hello,I\'m Son %s' % son_name) super().__init__('Neo\'s_father')if __name__ == '__main__': son = Son('Neo') # >>> 执行结果如下:# >>> Hello,I'm Son Neo# >>> Hello,I'm Father Neo's_father# >>> 执行结果得出:supper() 函数 很好的兼容了 Python 2.x 与 Python 3.x 的版本。
类的多重继承
什么是多重继承?为什么使用多重继承?
继承一词我们已经不陌生了,通过继承父类,我们可以在子类中使用父类的属性和方法。而多重继承,可以让子类同时继承多个父类,在实际工作中很多我们需要使用的函数、方法被分散放在多个类中。
如果想要在同一个类中使用这些被分散的各个方法,就需要在子类中继承多个父类,这样的话,我们就可以集中在一个子类中使用多个父类的方法了。
多重继承的使用方法
使用方法示例如下:
class Son(Father01, Father02, Father03... )# >>> 将被继承的多个父类放入子类小括号的参数位,每个父类使用逗号隔开# >>> 继承顺序,从左向右依次继承的
演示案例如下:
# 定义两个父类: Father() 类 与 Mother() 类class Father(object): def play_game(self): print('爸爸喜欢打游戏') def bodybuilding(self): print('爸爸喜欢的健身项目是跑步')class Mother(object): def buy(self): print('妈妈喜欢逛淘宝') def bodybuilding(self): print('妈妈喜欢的健身项目是瑜伽')# 继承 Father() 类 与 Mother() 类 的子类,使用 pass 做占位,不做具体业务处理class Son(Father, Mother): passif __name__ == '__main__': son = Son() son.play_game() son.buy() son.bodybuilding() # >>> 执行结果如下:# >>> 爸爸喜欢打游戏# >>> 妈妈喜欢逛淘宝# >>> 爸爸喜欢的健身项目是跑步# >>> 从执行结果来看,Son类 继承了 Father类与Mather类 后,可以同时使用 Father类 与 Mather类 的函数# >>> 同时我们也发现,Father类与Mather类都具有相同的函数 bodybuilding() ,当我们的 Son类执行 bodybuilding() 函数时,# >>> 优先执行的是第一个传入父类 Father类 的 bodybuilding() 函数,# >>> 说明继承的多个父类中存在相同函数时,是按照顺序从左向右依次继承,其中最左侧的类中的函数才会发生作用。
拓展:如何查看类的继承顺序
内置函数: __mro__
class Father(object): def play_game(self): print('爸爸喜欢打游戏') def bodybuilding(self): print('爸爸喜欢的健身项目是跑步')class Mother(object): def buy(self): print('妈妈喜欢逛淘宝') def bodybuilding(self): print('妈妈喜欢的健身项目是瑜伽')class Son(Father, Mother): passif __name__ == '__main__': print('Son类的继承顺序为:', Son.__mro__) # >>> 执行结果如下:# >>> Son类的继承顺序为:(, , , )