Python的多重继承和MixIn

2023-09-22 14:15:00

前言:

嗨喽~大家好呀,这里是魔王呐 ❤ ~!

python更多源码/资料/解答/教程等 点击此处跳转文末名片免费获取

多重继承

继承是面向对象编程的一个重要的方式,因为通过继承,子类就可以扩展父类的功能。

回忆一下Animal类层次的设计,假设我们要实现以下4种动物:

  • Dog - 狗狗;

  • Bat - 蝙蝠;

  • Parrot - 鹦鹉;

  • Ostrich - 鸵鸟。

如果按照哺乳动物和鸟类归类,我们可以设计出这样的类的层次:

但是如果按照“能跑”和“能飞”来归类,我们就应该设计出这样的类的层次:

如果要把上面的两种分类都包含进来,我们就得设计更多的层次:

  • 哺乳类:能跑的哺乳类,能飞的哺乳类;

  • 鸟类:能跑的鸟类,能飞的鸟类。

这么一来,类的层次就复杂了:

如果要再增加“宠物类”和“非宠物类”,这么搞下去,类的数量会呈指数增长,很明显这样设计是不行的。

正确的做法是采用多重继承。首先,主要的类层次仍按照哺乳类和鸟类设计:

class Animal(object):
    pass

# 大类:
class Mammal(Animal):
    pass

class Bird(Animal):
    pass

# 各种动物:
class Dog(Mammal):
    pass

class Bat(Mammal):
    pass

class Parrot(Bird):
    pass

class Ostrich(Bird):
    pass

现在,我们要给动物再加上Runnable和Flyable的功能,只需要先定义好Runnable和Flyable的类:

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:926207505
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
class Runnable(object):
    def run(self):
        print('Running...')

class Flyable(object):
    def fly(self):
        print('Flying...')

对于需要Runnable功能的动物,就多继承一个Runnable,例如Dog:

class Dog(Mammal, Runnable):
    pass 

对于需要Flyable功能的动物,就多继承一个Flyable,例如Bat:

class Bat(Mammal, Flyable):
    pass

通过多重继承,一个子类就可以同时获得多个父类的所有功能。

MixIn

在设计类的继承关系时,通常,主线都是单一继承下来的,例如,Ostrich继承自Bird。

但是,如果需要“混入”额外的功能,通过多重继承就可以实现,比如,让Ostrich除了继承自Bird外,再同时继承Runnable。

这种设计通常称之为Mixin。

为了更好地看出继承关系,我们把Runnable和Flyable改为RunnableMixin和FlyableMixin。

类似的,你还可以定义出肉食动物CarnivorousMixin和植食动物HerbivoresMixin,让某个动物同时拥有好几个Mixin:

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:926207505
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
class Dog(Mammal, RunnableMixin, CarnivorousMixin):
    pass

Mixin的目的就是给一个类增加多个功能,这样,在设计类的时候,我们优先考虑通过多重继承来组合多个Mixin的功能,而不是设计多层次的复杂的继承关系。

Python自带的很多库也使用了Mixin。

举个例子:

Python自带了TCPServer和UDPServer这两类网络服务,而要同时服务多个用户就必须使用多进程或多线程模型,这两种模型由ForkingMixin和ThreadingMixin提供。

通过组合,我们就可以创造出合适的服务来。

比如,编写一个多进程模式的TCP服务,定义如下:

class MyTCPServer(TCPServer, ForkingMixin):
    pass

编写一个多线程模式的UDP服务,定义如下:

class MyUDPServer(UDPServer, ThreadingMixin):
    pass

如果你打算搞一个更先进的协程模型,可以编写一个CoroutineMixin:

class MyTCPServer(TCPServer, CoroutineMixin):
    pass

这样一来,我们不需要复杂而庞大的继承链,只要选择组合不同的类的功能,就可以快速构造出所需的子类。

小结

由于Python允许使用多重继承,因此,Mixin就是一种常见的设计。

只允许单一继承的语言(如Java)不能使用Mixin的设计。

和多重继承类似(其实可以把 Mixin 看作多重继承的一种在特定场景下的应用),但通常混入 Mixin 的类和 Mixin 类本身不是 is-a 的关系,混入 Mixin 类是为了添加某些(可选的)功能。

自由地混入 Mixin 类就可以灵活地为被混入的类添加不同的功能。

尾语

最后感谢你观看我的文章呐~本次航班到这里就结束啦 🛬

希望本篇文章有对你带来帮助 🎉,有学习到一点知识~

躲起来的星星🍥也在努力发光,你也要努力加油(让我们一起努力叭)。

最后,宣传一下呀~👇👇👇更多源码、资料、素材、解答、交流皆点击下方名片获取呀👇👇

更多推荐

Palantir的“英伟达时刻”即将到来

来源:猛兽财经作者:猛兽财经总结(1)由于投资者对生成式人工智能的兴趣持续增加,Palantir的股价一直在上涨。(2)Palantir已经连续三个季度实现了GAAP盈利,并将很快有资格被纳入标普500指数。(3)Palantir拥有非常健康的资产负债表,并授权了一项股票回购计划。(4)虽然市场已经消化了很多乐观情绪,

RocketMQ高性能核心原理与源码架构剖析

文章目录1、源码环境搭建1.1、主要功能模块1.2、源码启动服务1.2.1、启动nameServer1.2.2、启动Broker1.2.3、发送消息1.2.4、消费消息1、源码环境搭建1.1、主要功能模块​RocketMQ的官方Git仓库地址:https://github.com/apache/rocketmq可以用g

【数据结构初阶】三、 线性表里的链表(无头+单向+非循环链表)

=========================================================================相关代码gitee自取:C语言学习日记:加油努力(gitee.com)====================================================

读高性能MySQL(第4版)笔记10_查询性能优化(上)

1.三管齐下1.1.不做、少做、快速地做1.2.如果查询太大,服务端会拒绝接收更多的数据并抛出相应错误1.3.如果查询写得很糟糕,即使库表结构再合理、索引再合适,也无法实现高性能1.4.查询优化、索引优化、库表结构优化需要齐头并进,一个不落1.5.PerconaToolkit中的pt-archiver工具2.响应时间2

【vivo秋招0912】三、最少开发工时总和 <模拟>

三、最少开发工时总和某开发小组近期承接了多个研发项目,作为组长的你需要为员工分配工作任务。具体要求如下:项目划分到的任务工时用二维数组tasks表示,其中tasks[i][j]表示的是第i个项目中第j个任务的开发工时;现在组内员工有n个,每个工作任务只能分配给一位员工,一位员工可以被分配多个任务,一个任务完成才能进行下

【用unity实现100个游戏之12】unity制作一个俯视角2DRPG《类星露谷物语》资源收集游戏demo

文章目录前言加快编辑器运行速度素材(1)场景人物(2)工具一、人物移动和动画切换二、走路灰尘粒子效果探究实现三、树木排序设计方法一方法二四、绘制拿工具的角色动画五、砍树实现六、存储拾取物品引入Unity的可序列化字典类七、实现靠近收获物品自动吸附八、树木被砍掉的粒子效果九、新增更多可收集物十、更多工具切换十一、扩展源码

第五天:基于gec6818开发板或stm32驱动对RFID高频读卡器进行防碰撞获取卡号,读取,写入内容操作

在ARM板上通过防碰撞指令获取高频RFID卡的卡号高频RFID一般频率在13.56MHz左右,识别距离可扩展至1.5米,具有防碰撞特性,可以同时读取多个电子标签。标签可以存储少量数据,多用于需要存储数据、近距离的识别应用,如:图书馆管理系统、场地通道安全管理系统等。mifare522模块的数据帧格式gec6818开发板

[C++ 网络协议] I/O流分离所带来的半关闭问题

1.问题和解决方法根据所学内容,I/O流分离现如今有如下2种方法:1.调用进程fork函数,分离出子进程,主进程和子进程分别进行输入流的读和输出流的写。2.用FILE指针按读模式和写模式将输入流和输出流进行区分。第一种方法,可以对输出流调用shutdown函数进行半关闭,从而不会影响到输入流接收客户端的数据,这是没问题

从理解js双重递归执行顺序到用递归方式实现二叉树中序遍历

今天在学习力扣上94题二叉树的中序遍历时,js的实现方法之一是递归,但是函数内递归是双重,花了一些时间来理解双重递归调用的执行顺序。先看如下例子,参考文章(双递归的执行过程理解)示例代码如下:constfn=(n)=>{if(n>0){console.log('n1====',n)fn(n-1)console.log(

微软AIGC in a Day-探索人工智能与行业应用实践沙龙-参后感

先来看下宣传海报活动介绍总结活动主题:探索人工智能与行业应用实践沙龙微软PowerPlatform携手GPT,从应用层面深入AI+低代码开发,一场探索人工智能与行业应用实践的技术盛宴即将到来!9月16日,「探索人工智能与行业应用实践沙龙」,数位来自AI、低代码领域的技术专家,将通过**技术分享、案例实操等形式,带来AI

美国调查公司 Digital Discovery 利用OpenText Encase 调查取证工具发现隐藏在数据中的事实

美国调查公司依靠OpenTextEnCase来分析数据挑战备受瞩目的复杂案件需要全面的取证解决方案;需要快速确定关键文档和证据的优先级;需要最新的支持来访问隐藏在多个来源和格式中的数据。结果为客户提供强大的取证调查解决方案;加速可辩护证据收集和分析;确保为最广泛的设备、格式和系统提供最新支持。故事DigitalDisc

热文推荐