Python零基础入门篇 - 37 - 类的继承、多态与多重继承(1)

  • 当我们定义了一个新的 类 时,可以从当前存在的 类 通过继承的关系得到其功能、属性等;被继承的 类 我们就叫做父类(也可以叫做基类或者超类),而新的 类 则叫做子类,且其具备父类的功能、属性与一些特性。

父类与子类

  • 首先,子类继承了父类,那么子类就拥有了父类的所有属性、方法。实现了代码的重用,那么相同的代码不需要重复编写,极大的提高了代码的可扩展性和重用性

  • 子类在通过继承拥有了父类的所有属性、方法后,也可以编写仅属于自己的新属性、新方法等,与父类并不冲突。父类不具备子类自有的属性与方法

如何使用继承

我们来看一个例子:

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)

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里无偿获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 22
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值