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

来点简单的,Adaptor 适配器模式

阅读更多
今天看了黑暗浪子的博客,喜欢他的博客,一看就是个爽快人。很高兴,一高兴,就想起来贴篇自己的文章以示高兴。

封装思想的又一种应用。类似挂羊头卖狗肉。比如一个类业已存在(类A),且正合我用,唯一遗憾是该类的方法名与用户要求的不同,那么用另一个新的、方法名符合规定的类套在类A身上是最顺其自然不过的方法了。新类的所作所为无非都是通过调用类A而实现。

在什么场合才需要挂羊头卖狗肉。第一,用户不关心是羊肉还是狗肉,只要是肉就行;第二,你手头没有羊肉而有现成狗肉。第三,用户需要羊头做为通行证。

这是类比,类比形象,但容易误导;所以贴出类比的原型,它是《DesignPatternsExplained》中的一个例子。这个例子是这样的,

有3个类,点、线、正方形。它们都有一个display方法。客户端不关心图形类型,只知道它们是图形,能画出自己来。所以,这里有一个抽象类,叫Shape,它有一个方法叫display. 原来的3个类都来继承Shape. 客户端只知道这个父类 (类比为肉),而不知什么点、线、方块。根据多态,此时Shape类可能是点的实例,也可能是线或正方形的实例,因此它所能画的图形类型也就这三种。如果此时多了一种需求,想让它也能画圆,那么就需要多写一个继承自Shape的Circle类负责画圆。摩拳擦掌准备再展弹指神功之际,突然发现有一个现成的类CircleA(狗肉)可以负起画圆的重任,且此类已经千锤百炼,已应用到好多模块中去了。现成的自然要拿来,问题是让此类直接继承Shape类是不可能的,一来CircleA的接口和Shape类要求继承实现的接口并不匹配,二来,重新修改Circle类会影响到其它已使用该类的模块。所以,在既想使用现成的类,又要匹配Shape的接口的情况下,自然就会创建一个新类,它对外继承了Shape接口(即,客户端要求的羊头,实际是一种契约,一种通行证),对内实际调用Circle类(狗肉,并无贬义)。这个新的类就是适配器。详细的图见下:

http://hiphotos.baidu.com/dapplehou/pic/item/640e7d1eb87dbdc51ad57600.jpg

适配器模式的实现,一般来说有两种方式;一种是用适配器对象包含另一个已经存在的对象,比如图中的Circle 包裹着  XXCircle;另一种方式是通过多继承的方式,比如Circle类可以不去调用XXCircle,而是继承XXCircle,同时再继承Shape(前提是Shape是个接口),既,Circle extends XXCircle implements Shape。它一样可以达到使用现成的类,却仍能匹配用户接口的目的(挂羊头卖狗肉)。

适配器模式也是一种封装,重点强调的接口转置,Façade模式也是一种封装,重点强调的是接口的简化。封装何其常见哉,比如我们写的类也会被别的模块调用,而我们写的类里一定使用了更通用的类,起码会用到基础类库,所以,我们的类一定也不知不觉的进行了封装工作,很可能我们的一个封装里既有对系统接口的简化,又有对部分已有接口的转置,那么这种封装该叫什么模式?所以说,不要拘泥于具体的招数,具体的模式皆是剑招,思想才是剑意。

就Adapter模式而言,你完全可以让这个Adapter除了做转置接口的工作外,再让它实现一些有助于解决实际问题的功能。这就如同练拳时野马分鬃右绷作捋,一开一合,潇洒无限,实战中难道因为我要遵守招式正统,而就不允许分鬃之余顺便吐他口痰么?不要为了模式而模式,约束了自己的创造力。

0
2
分享到:
评论
3 楼 Dapple 2010-06-22  
“不关心”是个有意思的经济学问题。
2 楼 mercyblitz 2010-06-22  
其实,很多API的利用都是多种模式的体现,非要指定那种模式的话,太狭隘了。

模式有很多,不应拘泥于GOF
1 楼 黑暗浪子 2010-06-22  
我只要我的笔记本有电,不关心我的电线插头是插在墙壁上的电源里还是“公牛”牌电源板上。哈哈~

相关推荐

    51丨适配器模式:代理、适配器、桥接、装饰,这四个模式有何区别?1

    适配器模式是一种结构型设计模式,其主要目的是解决接口不兼容问题,使得原本由于接口差异无法协同工作的类能够一起工作。适配器模式通过创建一个新的类(适配器),将旧的接口转换成目标接口,从而实现了系统的解耦...

    adaptor设计模式

    4. 协议适配:在Swift中,我们可以利用协议和扩展来实现适配器模式。通过为不满足某个协议的类提供一个遵循该协议的扩展,可以让类具备协议要求的能力。 下面是一个简单的适配器模式的示例: ```swift // 原始类,...

    0-30 isis_switching_Adaptor_3a_

    开关模式电源相比传统的线性电源,有更高的能效和更小的体积,因为它们通过高频开关操作来调节输出电压,而不是简单地依靠电阻降压。 在IT硬件中,这样的适配器可能被用在各种场景,例如: 1. **嵌入式系统**:...

    23种java版设计模式源码案例.zip

    适配器模式(adaptor) 桥接模式(bridge) 组合模式(composite) 装饰器模式(decorate) 外观模式(facecade) 享元模式(flyweight) 代理模式(proxy) 行为型模式(behaviour) 责任链模式(chainrespon) 命令模式(commond) ...

    homework for redis adaptor test

    【Redis适配器测试的家庭作业】是针对使用Redis作为数据存储和处理的系统进行的一种实践练习,主要目的是理解和掌握如何有效地使用Redis适配器来优化应用程序的性能和可扩展性。Redis是一个开源、高性能的键值数据库...

    二十三种设计模式【PDF版】

    2.设计模式是比 J2EE 等框架软件更小的体系结构,J2EE 中许多具体程序都是应用设计模式来完成的,当你深入到 J2EE 的内 部代码研究时,这点尤其明显,因此,如果你不具备设计模式的基础知识(GoF 的设计模式),你很难...

    超市管理系统java源码swing-king-design:23种java设计模式

    适配器模式(adaptor) 桥接模式(bridge) 组合模式(composite) 装饰器模式(decorate) 外观模式(facecade) 享元模式(flyweight) 代理模式(proxy) 行为型模式(behaviour) 责任链模式(chainrespon) 命令模式(commond) ...

    neo4j-core:一个简单的统一API,可以访问服务器和嵌入式Neo4j数据库。 由neo4j gem使用

    围绕Neo4j图形数据库的简单Ruby包装器,可与服务器和嵌入式Neo4j API配合使用。 可以在JRuby和普通MRI中使用该宝石。 它可以在没有neo4j gem的情况下单独使用。 基本用法 执行Cypher查询 要与Neo4j建立基本连接以...

    开源日志系统比较:scribe、chukwa、kafka、flume.docx

    - **Adaptor**:适配器模块,负责从各种数据源收集日志,支持Hadoop日志、应用度量和系统参数等。 - **Agent**:协调adaptor工作,通过HTTP将数据发送给Collector,同时记录adaptor状态以备恢复。 - **Collector**:...

    Jpos Programmer's Guide 161

    - **Task Adaptor**:任务适配器,用于处理异步任务。 - **DailyTask Adaptor**:用于定时执行的任务。 - **SMAdaptor**:系统监控适配器。 - **KeyStore Adaptor**:密钥存储管理。 - **QExec**:执行器组件。 - **...

    STL源码剖析

    5. 配接器(Adaptor):如stack(栈)、queue(队列)、priority_queue(优先队列)等,它们是基于现有容器构建的特定数据结构,提供了更简单的接口。 6. 适配器容器:例如,unordered_set和unordered_map是基于...

    ARM-Linux-ⅡC设备的添加与驱动实现.pdf

    - **适配器(Adaptor)**:针对不同的硬件平台,实现IIC总线控制器的具体操作。 - **算法(Algorithm)**:提供与特定IIC控制器交互的策略。 - **客户驱动(Client Driver)**:针对特定IIC设备的驱动程序,处理设备...

    将新的ATtiny处理器与Arduino IDE结合使用-项目开发

    "led programmer"标签暗示了至少有一个示例项目可能涉及LED的编程控制,可能是通过改变颜色或者闪烁模式来展示ATtiny1614的控制功能。 在压缩包的文件列表中,我们看到以下内容: 1. `colorchanger1614_ino.c`:这...

Global site tag (gtag.js) - Google Analytics