`
javayestome
  • 浏览: 1050673 次
  • 性别: Icon_minigender_2
  • 来自: 北京
文章分类
社区版块
存档分类
最新评论

[python] Mixin 扫盲班

阅读更多
Mixin 扫盲班
赖勇浩(http://blog.csdn.net/lanphaday)
声明:本文适合初中级Python程序员阅读,另外本文措词可能会导致阅读者不适,敬请PG。
引子
嗯,为什么要谈Mixin啊?
因为出现了Mixin这样一个东西呀,就像C++社区很多人谈论template一样啊,Python社区也很多会谈论Mixin的(以后会的,嘻嘻),所以我就来凑凑热闹了。
嗯,为什么要Mixin呀?
这个,基本上已经是我这篇文章里要讲的东西了,所以,我会用本文的大部分篇幅来回答你这个超有深度的问题。现在,就开始吧~
小时候,我家开百货店,当然,也兼营水果蔬菜啊。
小时候的事,总是特别有趣,也特别的有意义,所以今天我们就以水果店开始吧~
记得,以前很多人买水果的时候,都会问我妈一个问题,就是价格了啦~但还有两个问题也经常问到哦,就是送人应该买什么水果和什么水果可以用来送人?
嗯,年青人大多都不懂这些礼节。送水果也是很讲兆头的,比如梨和香蕉一般不用来送人,因为它们意味着离别和焦躁哦;而苹果和桔子就很受欢迎,因为它们意味着平安和吉利哦~
以此为开始
那,这跟Python有什么关系吗?
当然有啦,不然我扯那么多皮干嘛咧?现在,国产凌凌漆接到一个任务,要求他为一个水果连锁店开发一套软件;显然这个不关心国计民生这些鸡毛蒜皮的小事的武夫是搞不定这项艰巨任务的了,他就找到了你。
通过调研,客户发现两件事实:一是现在的年青人还是不懂送人应该买什么水果和什么水果可以用来送人这两个问题;二是水果连锁店的营业员100%都是年青人,他们也不懂。
所以,客户决定在软件中必须提供一个这样的功能--可以查询一种水果是否适宜送人。
最初,你可能这样设计:
class Fruit(object):
pass
把fruit类作为一切水果的基类,嗯,这相当明智。代码中去除了一些无需关注的代码,如价格、产地等。
现在你打算实现最受顾客欢迎的苹果:
class Apple(Fruit):
def is_gift_fruit(self):
return True
同样的,我又去除了一些无需关注的代码,并且打算在接下来的行文中不再提醒这一点。
Apple is a fruit。所以上面的实现挺符合OO的原则。
接下来让我们实现梨子吧:
class Pear(Fruit):
def is_gift_fruit(self):
return False
解决问题了。如果水果连锁店只卖苹果和梨子两种水果的话。
可惜,需求很多,你还要实现桔子和香蕉呢。你写下了这几行代码:
class Orange(Fruit):
def is_gift_fruit(self):
return True
class Banana(Fruit):
def is_gift_fruit(self):
return False
好臭啊,代码的坏味道!
类apple和类Orange除了类名不同,几乎是完全重复的代码;类Pear和类Banana也是一样。
更进一层的说,这四个类都差不多啊,所以我们有必要重构一下已有代码,改善它们的设计。
改善已有代码
阅读代码,你可以发现水果只分两类:一类是可以作为礼品的,一类是不可以的。所以希望可以这样设计:
Fruit
/ \
GiftFruitNotGiftFruit
/ \ / \
Apple Orange Pear Banana
嗯,加了两个中间类,看起来不错:
class GiftFruit(Fruit):
def is_gift_fruit(self):
return True
class NotGiftFruit(Fruit):
def is_gift_fruit(self):
return False
class Apple(GiftFruit):pass
class Orange(GiftFruit):pass
class Pear(NotGiftFruit):pass
class Banana(NotGiftFruit):pass
好啦,看上去很不错哦,代码精简了不少,任务完成~
新的烦恼
接下来我们来完成另一项功能:提供水果食用方法咨询。
不要笑这个需求,这是真实的市场需求。比如相当部分一辈子生活在北方的朋友就没有吃过龙眼荔枝香蕉;而南方虽然水果丰富,但不知道山竹等洋水果的也大有人在。我们这个水果连锁店业务简单,水果的食用方法也只分两种:一种是剥皮的,如桔子和香蕉;另一种是削皮的,如苹果和梨子。让我们修改原有的设计:
Fruit
/ \
GiftFruitNotGiftFruit
/ \ / \
PareG... HuskG... PareNot... HuskNot...
/ / / /
Apple Orange Pear Banana
不得已,我们添加了四个类:
class PareGiftFruit(GiftFruit):
def eat_method(self):
return 'Pare'
class HustGiftFruit(GiftFruit):
def eat_method(self):
return 'Husk'
class PareNotGiftFruit(NotGiftFruit):
def eat_method(self):
return 'Pare'
class HuskNotGiftFruit(NotGiftFruit):
def eat_method(self):
return 'Husk'
怎么这四个类这么像啊?汗。。。。
先忍忍,把AOPB四种水果的实现改改:
class Apple(PareGiftFruit):pass
class Orange(HuskGiftFruit):pass
class Pear(PareNotGiftFruit):Pass
class Banana(HuskNotGiftFruit):pass
我已经忍无可忍了。这个设计不仅仅又引入了好不容易消除的重复代码,而且还修改了AOPB这四个类的实现。这种设计的扩展性也不好,如果以后要提供水果的其它特点,比如是进口水果还是国产水果。天啊,这还了得!加上这个特性,我要实现LocalPareGiftFruit、LocalHuskGiftFruit等类共8个(2的三次方)啊。水果的特征多得很,随便算算可能超过16种啊,65536个类?叫我去死吧~单是长达16个单词的类名我就崩溃了!
现在,你们都应该意识到这种实现方法实在是一种龌龊的设计了。那,我们应该怎么样设计呢?
Pythonic的方案
该是Mixin出场的时候了!
先来看看Mixin的实现吧:
class Fruit(object):
pass
class GiftMixin(object):
def is_gift_fruit(self):
return True
class NotGiftMixin(object):
def is_gift_fruit(self):
return False
class PareMixin(object):
def eat_method(self):
return 'Pare'
class HuskMixin(object):
def eat_method(self):
return 'Husk'
class Apple(GiftMixin, PareMixin, Fruit):pass
class Orange(GiftMixin, HuskMixin, Fruit):pass
class Pear(NotGiftMixin, PareMixin, Fruit):pass
class Banana(NotGiftMixin, HuskMixin, Fruit):pass
编码完成!这就是Mixin,就是这么简单,以致我无法再说出任何言语,因为我觉得上面的代码已经完整地表达了我想要表达的思想。
Mixin的好处是可以为主类(如Fruit)添加任意多的Mixin来实现多态,比如刚才说的水果有进口和国产两个特征,现在相当容易实现:
class NativeMixin(object):
def Locality(self):
return 'Native'
class ForeignMixin(object):
def Locality(self):
return 'Foreign'
class Apple(ForeignMixin, GiftMixin, PareMixin, Fruit):pass #进口红富士
class Orange(NativeMixin, GiftMixin, HuskMixin, Fruit):pass
class Pear(NativeMixin, NotGiftMixin, PareMixin, Fruit):pass
class Banana(NativeMixin, NotGiftMixin, HuskMixin, Fruit):pass
简单多了,只加了两个类,对AOPB的实现也只是增加了一个基类(增加总是胜过修改)。
利用Mixin我们还可以增加无数总特征,而无需对已有代码作太大改动。
除此之外
这时候,你可能会说:水果连锁店软件只是你杜撰的一个项目,Mixin有什么实际用处吗?当然有啦!其实Mixin并不是什么高阶的Python技巧,早有就很多开源项目使用这个技巧了,典型的,比如Python项目啊!在Python自带的SimpleServer.py里就应用了Mixin来实现基于进程和基于线程的两种TCP/UDP服务模型,在Tkinter和Python的其它模块也可以见到它的踪迹,如果你留意的话。
确切来说,我对Mixin来实现的水果连锁店的实现仍然相当不满意,但如果我们想要足够面向对象,也就基本上只能接受如此解决方案了。如果有一天你不能忍受每增加一种特征你就必须编写N(N>=2)个Mixin,然后都必须给已经存在的AOPB代码增加一个基类(想想,水果店卖的可不止四种水果,你会更头大),那,就考虑把OO抛弃吧!
鸣谢
在本文成文过程中,沈崴(http://eishn.blog.163.com)给我很大帮助,特此鸣谢。
分享到:
评论

相关推荐

    使用Mixin设计模式进行Python编程的方法讲解

    Mixin模式是一种在python里经常使用的模式,适当合理的应用能够达到复用代码,合理组织代码结构的目的。 Python的Mixin模式可以通过多继承的方式来实现, 举例来说,我们自定义一个简单的具有嵌套结构的数据容器: ...

    Python库 | pickle-mixin-1.0.2.tar.gz

    Python库pickle-mixin-1.0.2是一个用于序列化和反序列化的工具,它扩展了Python内置的pickle模块。pickle模块是Python中用于将对象的状态转化为可存储或可传输格式,以及从这种格式恢复对象的原状的工具。这个库的...

    Python库 | colcon_mixin-0.1.3-py3-none-any.whl

    资源分类:Python库 所属语言:Python 资源全名:colcon_mixin-0.1.3-py3-none-any.whl 资源来源:官方 安装方法:https://lanzao.blog.csdn.net/article/details/101784059

    小程序mixin混入Page选项合并

    在微信小程序的开发中,`mixin` 是一种常见的代码复用机制,它允许开发者定义一组通用的方法或者属性,然后将这些通用部分混入(mix in)到不同的页面(Page)或组件(Component)中,避免了代码重复,提高了代码的...

    Mixin,MIXIN是使用ASM的Java的特性/混合框架.zip

    Mixin框架是Java编程语言中的一个开源项目,它利用ASM库来实现对类的低级别修改,也就是所谓的"混合"(Mixins)。这个框架的核心概念是将功能或行为注入到目标类中,而无需继承或者使用代理模式。在游戏开发、模块化...

    mixin-python3-sdk:用于python3.x的mixin SDK

    Mixin Python3 SDK 所有方法都在实现该SDK支持Python3.x 教程视频mixin_api.py 此SDK基于 您可以在test_api.py中看到演示mixin_ws_api.py 此SDK基于 您可以在ws_test.py中看到演示

    Python面向对象编程指南

    分别介绍了__init__()方法、与Python无缝集成—基本特殊方法、属性访问和特性及修饰符、抽象基类设计的一致性、可调用对象和上下文的使用、创建容器和集合、创建数值类型、装饰器和mixin—横切方面;第2部分讲述持久...

    ymixin是来自阅文前端团队的CSS预处理器mixin库

    【ymixin:阅文前端团队的CSS预处理器mixin库详解】 在前端开发中,CSS预处理器如Sass、Less和Stylus等已经成为提升样式编写效率和代码可维护性的必备工具。ymixin,作为阅文前端团队打造的一款CSS预处理器mixin库...

    Python中泛型函数应用案例.pdf

    Mixin是多重继承的一种形式,在Python中非常流行。文档中作者提到的StdIOMixin类就是一个典型的mixin类,它实现了标准输入输出的操作。在Mixin模式中,你通常创建一个只包含方法定义的类,并将这个类混入其他需要...

    mixin白皮书

    mixin是一个闪电交易快速的点点对的数字交道项目,它拥有非常好的技术栈。

    python 工程师技能图谱

    7. **高级Python特性**:了解Python的MRO算法,混合模式(Mixin)的使用,以及动态属性和特性属性的实现。深入理解元类和装饰器,它们在实现高级功能如缓存、日志记录、性能统计等方面非常有用。 8. **并发编程与...

    A Swift mixin for UITableViewCells and UICollectionViewCells.zip

    A Swift mixin for UITableViewCells and UICollectionViewCells.zip,A Swift mixin for reusing views easily and in a type-safe way (UITableViewCells, UICollectionViewCells, custom UIViews, ViewControllers...

    modernizr-mixin, 在Sass中,针对测试的简单而全面的mixin.zip

    modernizr-mixin, 在Sass中,针对测试的简单而全面的mixin hardwarebutton混合 一种简单的DRYier测试方法,在Sass中更快更。安装要求 ruby 3.4或者 LibSass 3.2Libsass警告:在 Libsass 3.2.3中有一个已知 Bug,它...

    公众号「Python编程时光」 干货目录.zip

    01. 基础系列1.1 基础课程1、盘点Python高手都写不出来的几个错误2、Python基础深入闭包与变量作用域3、Python基础类方法的强制重写与禁止重写4、Python基础多继承与Mixin设计模式5、Python基础理解元组存在的意义6...

    Python库 | mixer-6.2.5-py3-none-any.whl

    要在Python环境中安装mixin,首先确保已经安装了pip(Python的包管理器)。然后,使用以下命令安装从whl文件安装: ```bash pip install mixer-6.2.5-py3-none-any.whl ``` **使用mixin库:** 1. **导入mixin库:*...

    mixin_java_sdk:Mixin Client Java SDK

    Mixin Client Java SDK这里是 Mixin Client Java SDK,其它语言的 Mixin SDK:NodeJS:Go:Python:更多 Mixin 开发资源:mixin_dev_resource:MiXin_Player:Java SDK v0.2当前版本 v0.2,主要功能是 Mixin 机器人:...

    forge-mixin-example:在Minecraft Forge 1.12.2中使用Mixin的示例

    而Mixin是Java编程语言中的一种强大的代码注入框架,常用于Minecraft模组开发,它允许开发者无侵入地修改游戏的内部行为,而无需直接继承或修改原代码。本示例项目"forge-mixin-example"就是针对Minecraft Forge ...

    前端大厂最新面试题-mixin.docx

    Vue 中的 Mixin 及其应用场景 在 Vue 中,Mixin 是一种提供了方法实现的类,其他类可以访问 Mixin 类的方法而不必成为其子类。Mixin 类通常作为功能模块使用,在需要该功能时“混入”,有利于代码复用又避免了多...

Global site tag (gtag.js) - Google Analytics