`
yezongbo
  • 浏览: 30341 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

FlyintoMixin

阅读更多
1. 引子
嗯,为什么要谈Mixin啊?
因为出现了Mixin这样一个东西呀,就像C++社区很多人谈论template一样啊,Python社区也很多会谈论Mixin的(以后会的,嘻嘻),所以我就来凑凑热闹了。

嗯,为什么要Mixin呀?
这个,基本上已经是我这篇文章里要讲的东西了,所以,我会用本文的大部分篇幅来回答你这个超有深度的问题。现在,就开始吧~

小时候,我家开百货店,当然,也兼营水果蔬菜啊。

小时候的事,总是特别有趣,也特别的有意义,所以今天我们就以水果店开始吧~

记得,以前很多人买水果的时候,都会问我妈一个问题,就是价格了啦~但还有两个问题也经常问到哦,就是送人应该买什么水果和什么水果可以用来送人?

嗯,年青人大多都不懂这些礼节。送水果也是很讲兆头的,比如梨和香蕉一般不用来送人,因为它们意味着离别和焦躁哦;而苹果和桔子就很受欢迎,因为它们意味着平安和吉利哦~


1.1. 以此为开始
那,这跟Python有什么关系吗?
当然有啦,不然我扯那么多皮干嘛咧?现在,国产凌凌漆接到一个任务,要求他为一个水果连锁店开发一套软件;显然这个不关心国计民生这些鸡毛蒜皮的小事的武夫是搞不定这项艰巨任务的了,他就找到了你。

通过调研,客户发现两件事实:一是现在的年青人还是不懂送人应该买什么水果和什么水果可以用来送人这两个问题;二是水果连锁店的营业员100%都是年青人,他们中大部分人也不懂。

所以,客户要求在软件中必须提供一个这样的功能--可以查询一种水果是否适宜送人。


1.1.1. 最初
最初,你可能这样设计:

class Fruit(object):
pass
把fruit类作为一切水果的基类,嗯,这相当明智。代码中去除了一些无需关注的代码,如价格、产地等。
现在你打算实现最受顾客欢迎的苹果:

切换行号显示
   1 class Apple(Fruit):
   2  def is_gift_fruit(self):
   3   return True
同样的,我又去除了一些无需关注的代码,并且打算在接下来的行文中不再提醒这一点。
Apple是一种Fruit。所以上面的实现挺符合OO的原则。
接下来让我们实现梨子吧:

切换行号显示
   1 class Pear(Fruit):
   2  def is_gift_fruit(self):
   3   return False
解决问题了。如果水果连锁店只卖苹果和梨子两种水果的话。
可惜,需求很多,你还要实现桔子和香蕉呢。你写下了这几行代码:

切换行号显示
   1 class Orange(Fruit):
   2  def is_gift_fruit(self):
   3   return True
   4 class Banana(Fruit):
   5  def is_gift_fruit(self):
   6   return False
好臭啊,代码的坏味道!

类Apple和类Orange除了类名不同,几乎是完全重复的代码;类Pear和类Banana也是一样。 更进一层的说,这四个类都差不多啊,所以我们有必要重构一下已有代码,改善它们的设计。


1.1.2. 改善已有代码
阅读代码,你可以发现水果只分两类:一类是可以作为礼品的,一类是不可以的。所以希望可以这样设计:

  Fruit
  / \
GiftFruit NotGiftFruit
/ \ / \
Apple    Orange Pear Banana
嗯,加了两个中间类,看起来不错:

切换行号显示
   1 class GiftFruit(Fruit):
   2  def is_gift_fruit(self):
   3   return True
   4 class NotGiftFruit(Fruit):
   5  def is_gift_fruit(self):
   6   return False
   7 class Apple(GiftFruit):pass
   8 class Orange(GiftFruit):pass
   9 class Pear(NotGiftFruit):pass
  10 class Banana(NotGiftFruit):pass
好啦,看上去很不错哦,代码精简了不少,任务完成~


1.1.3. 新的烦恼
接下来我们来完成另一项功能:提供水果食用方法咨询。 表笑这个需求,这是真实的市场需求。比如相当部分一辈子生活在北方的朋友就没有吃过龙眼荔枝香蕉;而南方虽然水果丰富,但不知道山竹榴莲等洋水果的也大有人在。我们这个水果连锁店业务简单,水果的食用方法也只分两种:一种是剥皮的,如桔子和香蕉;另一种是削皮的,如苹果和梨子。让我们修改原有的设计:

   Fruit
   / \
  GiftFruit NotGiftFruit
  / \ /  \
PareG...   HuskG... PareNot... HuskNot...
/  / /  /
Apple     Orange  Pear  Banana
不得已,我们添加了四个类:

切换行号显示
   1 class PareGiftFruit(GiftFruit):
   2  def eat_method(self):
   3   return 'Pare'
   4 class HustGiftFruit(GiftFruit):
   5  def eat_method(self):
   6   return 'Husk'
   7 class PareNotGiftFruit(NotGiftFruit):
   8  def eat_method(self):
   9   return 'Pare'
  10 class HuskNotGiftFruit(NotGiftFruit):
  11  def eat_method(self):
  12   return 'Husk'
怎么这四个类这么像啊?汗。。。。
先忍忍,把AOPB四种水果的实现改改:

切换行号显示
   1 class Apple(PareGiftFruit):pass
   2 class Orange(HuskGiftFruit):pass
   3 class Pear(PareNotGiftFruit):Pass
   4 class Banana(HuskNotGiftFruit):pass
我已经忍无可忍了。这个设计不仅仅又引入了好不容易消除的重复代码,而且还修改了AOPB这四个类的实现。这种设计的扩展性也不好,如果以后要提供水果的其它特点,比如是进口水果还是国产水果。天啊,这还了得!加上这个特性,我要实现NativePareGiftFruit、NativeHuskGiftFruit等类共8个(2的三次方)啊。水果的特征多得很,随便算算可能超过16种啊,65536个类?叫我去死吧~单是长达16个单词的类名我就崩溃了!

现在,你们都应该意识到这种实现方法实在是一种龌龊的设计了。那,我们应该怎么样设计呢?

1.1.4. Pythonic的方案
该是Mixin出场的时候了! 先来看看Mixin的实现吧:

切换行号显示
   1 class Fruit(object):
   2  pass
   3 class GiftMixin(object):
   4  def is_gift_fruit(self):
   5   return True
   6 class NotGiftMixin(object):
   7  def is_gift_fruit(self):
   8   return False
   9 class PareMixin(object):
  10  def eat_method(self):
  11   return 'Pare'
  12 class HuskMixin(object):
  13  def eat_method(self):
  14   return 'Husk'
  15 class Apple(GiftMixin, PareMixin, Fruit):pass
  16 class Orange(GiftMixin, HuskMixin, Fruit):pass
  17 class Pear(NotGiftMixin, PareMixin, Fruit):pass
  18 class Banana(NotGiftMixin, HuskMixin, Fruit):pass
编码完成!这就是Mixin,就是这么简单,以致我无法再说出任何言语,因为我觉得上面的代码已经完整地表达了我想要表达的思想。

注意, 因为 Python 里面多重继承时如果被调用的成员函数只存在于父类中,则按类声明的父类从左到右查找调用的, 所以主类被放在右边, MixIn 被放在左边,才能正确地调用到Mixin的成员函数。


1.1.5. Mixin的好处
Mixin的好处是可以为主类(如Fruit)添加任意多的Mixin来实现多态,比如刚才说的水果有进口和国产两个特征,现在相当容易实现:

切换行号显示
   1 class NativeMixin(object):
   2  def Locality(self):
   3   return 'Native'
   4 class ForeignMixin(object):
   5  def Locality(self):
   6   return 'Foreign'
   7 class Apple(ForeignMixin, GiftMixin, PareMixin, Fruit):pass #进口红富士
   8 class Orange(NativeMixin, GiftMixin, HuskMixin, Fruit):pass
   9 class Pear(NativeMixin, NotGiftMixin, PareMixin, Fruit):pass
  10 class Banana(NativeMixin, NotGiftMixin, HuskMixin, Fruit):pass
简单多了,只加了两个类,对AOPB的实现也只是增加了一个基类(增加总是胜过修改)。
利用Mixin我们还可以增加无数总特征,而无需对已有代码作太大改动。
另外,我们还获得了可重用性。比如NativeMixin和ForeignMixin跟主类Human结合,可以做出国人和老外两个类哦~也许水果连锁店软件以后会考虑记录关于客户是否外国人的信息呢。


1.1.6. 除此之外
这时候,你可能会说:水果连锁店软件只是你杜撰的一个项目,Mixin有什么实际用处吗?
当然有啦!其实Mixin并不是什么高阶的Python技巧,早有就很多开源项目使用这个技巧了,典型的,比如Python项目啊!在Python自带的SocketServer.py里就应用了Mixin来实现基于进程和基于线程的两种TCP/UDP服务模型,在Tkinter和Python的其它模块也可以见到它的踪迹,如果你留意的话。


# SocketServer.py 里的Mixin
class ForkingUDPServer(ForkingMixIn, UDPServer): pass
class ForkingTCPServer(ForkingMixIn, TCPServer): pass
class ThreadingUDPServer(ThreadingMixIn, UDPServer): pass
class ThreadingTCPServer(ThreadingMixIn, TCPServer): pass
确切来说,我对Mixin来实现的水果连锁店的实现仍然相当不满意,但如果我们想要足够面向对象,也就基本上只能接受如此解决方案了。
如果有一天你不能忍受每增加一种特征你就必须编写N(N>=2)个Mixin,然后都必须给已经存在的AOPB代码增加一个基类(想想,水果店卖的可不止四种水果,你会更头大),那,就考虑把OO抛弃吧!


鸣谢
在本文成文过程中,
沈崴(http://eishn.blog.163.com)给我很大帮助,特此鸣谢。

2. 反馈

2.1. 原稿泄漏版本

切换行号显示
   1 #!/usr/bin/env python
   2 # 仿黄毅大师的版本
   3
   4 class Instance:
   5        def __init__(self, *args, **kw):
   6                self.__dict__.update(kw)
   7                for m in args:
   8                        m(self)
   9
  10        def config(self, *args, **kw):
  11                self.__dict__.update(kw)
  12                for m in args:
  13                        m(self)
  14
  15                return self
  16
  17 def i_am_gift(self):
  18        # self.is_gift = True # Why not
  19
  20        self.is_gift = lambda: True
  21
  22 def i_am_not_gift(self):
  23        self.is_gift = lambda: False
  24
  25 def eatable(eat_method = ''):
  26        def config_eat_method(self):
  27                self.eat_method = lambda: eat_method
  28
  29        return config_eat_method
  30
  31 def Apple():
  32        return Instance(i_am_gift, eatable('Bare'))
  33
  34 def Banana():
  35        return Instance(i_am_not_gift, eatable('Hust'))
  36
  37 if __name__ == '__main__':
  38        apple = Apple()
  39        print apple.is_gift()
  40
  41        apple.config(i_am_not_gift)
  42        print apple.is_gift()
  43
  44        banana = Banana()
  45        print apple.eat_method()
  46        print banana.eat_method()
分享到:
评论

相关推荐

    只需要用一张图片素材文档选择器.zip

    只需要用一张图片素材文档选择器.zip

    浙江大学842真题09-24 不含答案 信号与系统和数字电路

    浙江大学842真题09-24 不含答案 信号与系统和数字电路

    无标题baci和jbaci

    无标题baci和jbaci

    完整的雷达系统仿真程序,完整的雷达系统仿真程序 matlab代码.rar

    1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。 替换数据可以直接使用,注释清楚,适合新手

    实体商品销售源码最新优化.zip

    实体商品销售源码最新优化.zip

    戴尔存储MD1400机柜维护操作与安全指导

    内容概要:本文档详细介绍了戴尔存储MD1400机柜的安全注意事项、电源指示灯解释、故障排除方法以及硬件维护步骤,包括卸下和安装直流电源设备、硬盘驱动器和背板的具体操作流程。 适用人群:IT运维人员、数据中心管理员和技术支持工程师。 使用场景及目标:在维护和管理戴尔存储MD1400机柜时作为参考指南,确保正确安装和故障排查,避免安全隐患和设备损坏。 其他说明:文档提供了丰富的图文指导,帮助使用者更好地理解和执行相关操作。

    PyClass 课程计划.zip

    PyClass 课程计划Noisebridge Python 课程每周一晚上 7 点至 9 点(太平洋时间)在旧金山 Noisebridge 二楼电子室举行。自 2024 年 8 月起,该课程目前暂停。请参阅 wiki 页面了解更多信息。本课程免费!如果您希望捐款,请捐赠给 Noisebridge。建议捐款15 美元、50 美元、200 美元以上建议每月捐款每月 10 美元、20 美元、40 美元、80 美元以上所有 Python 课程均遵循Noisebridge 反骚扰政策、 Noisebridge 冲突解决指南和 recurse.org 社交规则课后,我们欢迎您提供反馈! 在此提交表格内容课程课程描述新生阅读迭代次数Noisebridge Python 课程至少早在 2015 年就已经存在,拥有许多不同的讲师和版本。从 2017 年到 2018 年,该课程似乎由Jared Garst负责。(?)。从 2023 年到 2024 年,该课程由Travis Briggs负责。如果您有其他关于此类历史的信息想要分享,请在此处创建 PR、

    自动化部署管道创建的代码库(含 Concourse 和 Jenkins 相关).zip

    1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。

    一种新的混合优化算法,即瞬态三角哈里斯鹰优化器(Tthho) matlab代码.rar

    1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。 替换数据可以直接使用,注释清楚,适合新手

    1-中国各地万达广场地理分布数据2006-2021-社科数据.zip

    万达广场作为城市综合体的代表,在中国各地的地理分布数据集覆盖了2006至2021年。这些数据详细记录了万达广场的多个关键指标,包括项目名称、项目信息、具体地点、开业时间、商业面积以及精确的经度和纬度。万达广场不仅是商业地产开发的先行者,还以其成熟的商业模式、完善的产业链和丰富的商业资源,在全国范围内形成了独立的大型商圈。这些综合体集购物、餐饮、文化、娱乐等多种功能于一体,对提升城市商业档次、增加就业岗位、创造税收以及丰富群众消费需求等方面产生了显著的社会效益。数据集提供了420条样本,为研究中国区域经济发展特征及其未来趋势提供了宝贵的信息资源。

    正在月下弹琴的古装美女flash场景动画.zip

    正在月下弹琴的古装美女flash场景动画.zip

    理光Ricoh-MP C8003打印机驱动下载

    理光 MP C8003 是一款彩色激光多功能数码复合机。 【基础性能】 打印复印速度:黑白和彩色打印 / 复印速度均可达 80 页每分钟,能够快速高效地完成大量文档的输出任务,有效提高工作效率 分辨率:拥有 1200x4800dpi 的高分辨率,可输出色彩鲜艳、细节丰富、图像清晰的文档和图像,满足专业级的打印和复印需求,尤其适合对色彩精度要求较高的设计图纸、宣传资料等文件的输出 首张输出时间:黑白首张复印时间为 4.7 秒,彩色首张复印时间为 6.3 秒,在启动打印或复印任务时无需长时间等待,可迅速响应,进一步提升工作效率 纸张容量:标准配置的纸张容量为单 2500 页抽屉和双 550 页抽屉,还可通过扩展将纸张容量从 3700 页提升至 8100 页,能够满足不同规模的打印任务需求,减少纸张添加的频率 【功能多样性】 多功能一体:集复印、打印、扫描、传真功能于一身,可满足办公室多样化的文档处理需求,一台设备即可替代多台单一功能的设备,节省空间和成本 扫描功能:具备高速扫描能力,可通过多种扫描至选项将原件扫描并以电子形式分发,支持将扫描后的文件直接发送至个人移动设备

    《The Annotated Transformer》环境配置

    《The Annotated Transformer》环境配置

    基于深度学习resnet50和vgg16卷积神经网络的汉字书法识别项目源码+训练集+测试集 【可用于课设-毕设】

    深度学习大作业基于resnet50和vgg16卷积神经网络的汉字书法识别项目源码+训练集+测试集 操作步骤 将下载的训练集和测试集,解压到工程中 运行对应的data.py文件,进行转录,将原始数据集转录为numpy矩阵,生成data.npy及label.npy 运行对应的train.py进行训练 运行test.py使用训练完成的网络测试。

    直接序列扩频(DSSS) matlab代码.rar

    1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。 替换数据可以直接使用,注释清楚,适合新手

    代码

    代码

    ECharts柱状图-基础柱状图.rar

    图表效果及代码实现讲解链接:https://blog.csdn.net/zhangjiujiu/article/details/143996614

    Spring Data Key Value 特性的示例项目.zip

    1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。

    营销策划 -魔術絲-黑松露香蕉莓果饮品-新品发布会产品介绍-终版.pptx

    营销策划 -魔術絲-黑松露香蕉莓果饮品-新品发布会产品介绍-终版.pptx

    成都市数据条例.docx

    成都市数据条例.docx

Global site tag (gtag.js) - Google Analytics