文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

python中Mixin混入类如何使用

2023-07-02 18:37

关注

这篇文章主要讲解了“python中Mixin混入类如何使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“python中Mixin混入类如何使用”吧!

前言

最近在看sanic的源码,发现有很多Mixin的类,大概长成这个样子

class BaseSanic(    RouteMixin,    MiddlewareMixin,    ListenerMixin,    ExceptionMixin,    SignalMixin,    metaclass=SanicMeta,):

于是对于这种 Mixin 研究了一下,其实也没什么新的东西,Mixin 又称混入,只是一种编程思想的体现,但是在使用过程中还是有一些需要注意的地方。 大家都知道,python 是一种允许多继承的语言,一个类可以继承多个类,这和java不太一样,java类只能有一个父类, 但是java 中有接口的概念,一个类可以实现多个接口,但是java的接口只是定义的函数的签名,并没有具体的实现,具体的实现需要相应的类来完成。 python 就不一样了,一个类可以有多个父类,而混入类就是这种允许多继承语言中才有的一种编程模式。 为了更好的理解混入,我们举一个生活中的例子---手机, 手机有很多功能,由不同的硬件组合而成,手机有接打电话,收发短信,上网,听歌等功能,组装一台手机就需要将各种硬件进行拼接。 如果我们把这些功能抽象成类,那么我们可以有以下写法,为了简单一点,只列接打电话,收发短信功能。

class Tel:    def telfunc(self):        print("我可以接打电话")        class SMS:    def smsfunc(self):        print("我可以发短信")        class Phone:    def __init__(self, sn):        self.sn = sn

上面的代码中, 有三个类,Tel 类,它有一个 telfunc 方法用于表示有接打电话的能力(或者说是功能), SMS 类有smsfunc表示SMS类有发短信的能力。 而 Phone 这个类才是一个手机类,它应该具有接打电话和发送短信的能力,但是如果我们用上面的方式定义Phone 这个类,则这个类并没有接打电话和收发短信的能力。 我们可以怎样做让Phone这个类可以具有打电话和发短信的能力? 我们可以在Phone 这个类里再重新定义二个方法 telfunc和 smsfunc,也就是将Tel类和SMS类里的方法再写一遍,这种其实不符合 don't repeate youself的理念。 正常情况下我们是让Phone这个类继承Tel类和SMS类,这样Phone这个类就自动拥有了接打电话和发短信的能力了。

class Tel:    def telfunc(self):        print("我可以接打电话")class SMS:    def smsfunc(self):        print("我可以发短信")class Phone(Tel, SMS):    def __init__(self, sn):        self.sn = sn    def welcome(self):        print("welcome {}".format(self.sn))p = Phone("xiaomi")p.telfunc()p.smsfunc()p.welcome()'''我可以接打电话我可以发短信welcome xiaomi'''

像这种类的定义就是我们所说的混入,将通话功能与短信功能加入到手机中,让手机拥有接打电话和发送短信的功能,这种混入的编码思想有时可以减少很多代码量。很方便的根据一个类需要哪些功能就将哪个类“混入”到该类中。 通常情况下,我们会将混入类的命名以Mixin结尾,像上面的代码我们会写成下面这样

class TelMixin:    def telfunc(self):        print("我可以接打电话")class SMSMixin:    def smsfunc(self):        print("我可以发短信")class Phone(TelMixin, SMSMixin):    def __init__(self, sn):        self.sn = sn    def welcome(self):        print("welcome {}".format(self.sn))

以 Mixin 结尾的类,一般是那种功能比较单一,且一般都是某一类型的功能, 如最开始介绍的sanic混入类 RouteMixin 与路由相关的功能类,MiddlewareMixin 是与中间件相关的类,ListenerMixin 是监听器相关的类,这些类里的方法专注于自己相关的功能,如果有哪个类需要这些功能,那定义的时候就继承自这些类。

有人会问了,我不想多继承,管理MRO 太麻烦了,我只想单继承,我定义一个统一的父类,它即有接打电话,也有收发短信的功能,还可以听歌,然后让手机来继承这个类不好吗?

如下面的代码:

class Tel:    def telfunc(self):        print("我可以接打电话")    def smsfunc(self):        print("我可以发短信")    def songfunc(self):        print("我可以放音乐")class Phone(Tel):    def __init__(self, sn):        self.sn = sn    def welcome(self):        print("welcome {}".format(self.sn))

首先这种写法当然是可以的,语法上没有任何问题,也很好的实现了相应的功能,代码量也减少了,但是这里大家想一个逻辑问题,如果我想造一个ipod类,ipod 这个类没有接打电话收发短信的功能,只有听歌的功能,那么我此时要写这个类该如何定义? 是不是要定义一个ipod类,然后再写它的听歌方法songfunc,如果此时使用混入的编程思想,那么我们就完全可以定义ipod类的时候 不加入 接打电话和收发短信的类就可以了。

class TelMixin:    def telfunc(self):        print("我可以接打电话")class SMSMixin:    def smsfunc(self):        print("我可以发短信")        class SongMixin:    def songfunc(self):        print("我可以放音乐")class Phone(TelMixin, SMSMixin, SongMixin):    def __init__(self, sn):        self.sn = snclass Ipod(SongMixin):    def __init__(self, sn):        self.sn = sn

如果还有别的什么类,比如对讲机,它只有接打电话的功能,那么我们就只需要把接打电话的功能混入到对讲机类即可, class Intercom(TelMixin):

但是有人又会问了,这样的混入,虽说少写了一些代码,但是如果子类相应实现并不完全和父类一致该如何处理? 如对讲机虽然可以通话,但是它是单向通话,并不能双向通话的。 这时对于子类实现与父类不一致的情况,那么就需要子类重写父类方法了,也就是OOP中的继承与多态,严格意义上来说, python 并没有多态,或者说它天然的就是多态。

class TelMixin:    def telfunc(self):        print("我可以接打电话")class SMSMixin:    def smsfunc(self):        print("我可以发短信")class SongMixin:    def songfunc(self):        print("我可以放音乐")class Intercom(TelMixin):    def __init__(self, sn):        self.sn = sn    def telfunc(self):        print("对讲机{} 的通话".format(self.sn))class Phone(TelMixin, SMSMixin, SongMixin):    def __init__(self, sn):        self.sn = snclass Ipod(SongMixin):    def __init__(self, sn):        self.sn = sndef testTelfunc(o):    o.telfunc()d = Intercom("moto")p = Phone("huawei")ipod = Ipod("apple")testTelfunc(d)testTelfunc(p)testTelfunc(ipod)

以上代码,我们重写了对讲机类的telfunc 方法,但是并没有对Ipod类实现通话功能, 然后写了一个testTelfunc(o) 方法来调用传入参数的通话功能,

得到以下输出:

#对讲机moto 的通话
#我可以接打电话
Traceback (most recent call last):
  File "/Users/yyx/test/test.py", line 54, in <module>
    testTelfunc(ipod)
  File "/Users/yyx/test/test.py", line 46, in testTelfunc
    o.telfunc()
AttributeError: 'Ipod' object has no attribute 'telfunc'

当给testTelfunc 传入的是对讲机Intercom 对象时,由于该类重写了telfunc方法,所以这里调用的是该子类的telfunc方法,当传入的是Phone类对象时,由于该类继承了TelMixin类,且没有重写telfunc方法,所以这里会调用TelMixin类中的telfunc 方法,但是当传入的是Ipod类时,这个类既没有继承TelMixin,也没有自己的telfunc方法,所以就崩溃了。这也是一种另类的多态体现吧,只是它没有像java中的那么严格。

Mixin 与继承的区别

说了那么多的Mixin混入,其实它本质上就是继承,只是这种继承是存在于允许多继承的编程语言中,如果说区别,本质上也没有什么区别,如果硬是要说些区别,其实也是有一点点。

说了那么多Mixin 的好处,它有什么不好的地方吗? 其实在我看来,有一个问题,也说不上是不好,只是习惯问题,以前写代码还是写单继承多一些,并不太习惯这种多继承的方式,不过如果一切都按照相应的规则来操作也不是什么大问题。 还有一个问题就是功能的拆分,这个很是考验编程人员在设计代码之初整体把握能力,就像单体应用想要拆分成微服务,边界问题可能是开始时最头疼的,拆细了吧,会写出很多类,拆粗了吧,就又回到单继承的思维模式了,所以这个也是个经验问题吧。

感谢各位的阅读,以上就是“python中Mixin混入类如何使用”的内容了,经过本文的学习后,相信大家对python中Mixin混入类如何使用这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯