一、引言

  在实际的开发过程中,由于应用环境的变化(例如使用语言的变化),我们需要的实现在新的环境中没有现存对象可以满足,但是其他环境却存在这样现存的对象。那么如果将“将现存的对象”在新的环境中进行调用呢?解决这个问题的办法就是我们本文要介绍的适配器模式——使得新环境中不需要去重复实现已经存在了的实现而很好地把现有对象(指原来环境中的现有对象)加入到新环境来使用。

二、适配器模式的详细介绍

2.1 定义

  下面让我们看看适配器的定义,适配器模式——把一个类的接口变换成客户端所期待的另一种接口,从而使原本接口不匹配而无法一起工作的两个类能够在一起工作。适配器模式有类的适配器模式和对象的适配器模式两种形式,下面我们分别讨论这两种形式的实现和给出对应的类图来帮助大家理清类之间的关系。

2.2 类的适配器模式实现

  在这里以生活中的一个例子来进行演示适配器模式的实现,具体场景是: 在生活中,我们买的电器插头是2个孔的,但是我们买的插座只有三个孔的,此时我们就希望电器的插头可以转换为三个孔的就好,这样我们就可以直接把它插在插座上,此时三个孔插头就是客户端期待的另一种接口,自然两个孔的插头就是现有的接口,适配器模式就是用来完成这种转换的,具体实现代码如下:

using System;
/// 这里以插座和插头的例子来诠释适配器模式
/// 现在我们买的电器插头是2个孔,但是我们买的插座只有3个孔的
/// 这是我们想把电器插在插座上的话就需要一个电适配器
namespace 设计模式之适配器模式
{/// <summary>/// 客户端,客户想要把2个孔的插头 转变成三个孔的插头,这个转变交给适配器就好/// 既然适配器需要完成这个功能,所以它必须同时具体2个孔插头和三个孔插头的特征/// </summary>class Client{static void Main(string[] args){// 现在客户端可以通过电适配要使用2个孔的插头了IThreeHole threehole = new PowerAdapter();threehole.Request();Console.ReadLine();}}/// <summary>/// 三个孔的插头,也就是适配器模式中的目标角色/// </summary>public interface IThreeHole{void Request();}/// <summary>/// 两个孔的插头,源角色——需要适配的类/// </summary>public abstract class TwoHole{public void SpecificRequest(){Console.WriteLine("我是两个孔的插头");}}/// <summary>/// 适配器类,接口要放在类的后面/// 适配器类提供了三个孔插头的行为,但其本质是调用两个孔插头的方法/// </summary>public class PowerAdapter:TwoHole,IThreeHole{/// <summary>/// 实现三个孔插头接口方法/// </summary>public void Request(){// 调用两个孔插头方法this.SpecificRequest();}}
}

从上面代码中可以看出,客户端希望调用Request方法(即三个孔插头),但是我们现有的类(即2个孔的插头)并没有Request方法,它只有SpecificRequest方法(即两个孔插头本身的方法),然而适配器类(适配器必须实现三个孔插头接口和继承两个孔插头类)可以提供这种转换,它提供了Request方法的实现(其内部调用的是两个孔插头,因为适配器只是一个外壳罢了,包装着两个孔插头(因为只有这样,电器才能使用),并向外界提供三个孔插头的外观,)以供客户端使用。

2.3 类图

上面实现中,因为适配器(PowerAdapter类)与源角色(TwoHole类)是继承关系,所以该适配器模式是类的适配器模式,具体对应的类图为:

2.4 对象的适配器模式

  上面都是类的适配器模式的介绍,然而适配器模式还有另外一种形式——对象的适配器模式,这里就具体讲解下它的实现,实现的分析思路:既然现在适配器类不能继承TwoHole抽象类了(因为用继承就属于类的适配器了),但是适配器类无论如何都要实现客户端期待的方法的,即Request方法,所以一定是要继承ThreeHole抽象类或IThreeHole接口的,然而适配器类的Request方法又必须调用TwoHole的SpecificRequest方法,又不能用继承,这时候就想,不能继承,但是我们可以在适配器类中创建TwoHole对象,然后在Requst中使用TwoHole的方法了。正如我们分析的那样,对象的适配器模式的实现正式如此。下面就让我看看具体实现代码:

namespace 对象的适配器模式
{class Client{static void Main(string[] args){// 现在客户端可以通过电适配要使用2个孔的插头了ThreeHole threehole = new PowerAdapter();threehole.Request();Console.ReadLine();}}/// <summary>/// 三个孔的插头,也就是适配器模式中的目标(Target)角色/// </summary>public class ThreeHole{// 客户端需要的方法public virtual void Request(){// 可以把一般实现放在这里
        }}/// <summary>/// 两个孔的插头,源角色——需要适配的类/// </summary>public class TwoHole{public void SpecificRequest(){Console.WriteLine("我是两个孔的插头");}}/// <summary>/// 适配器类,这里适配器类没有TwoHole类,/// 而是引用了TwoHole对象,所以是对象的适配器模式的实现/// </summary>public class PowerAdapter : ThreeHole{// 引用两个孔插头的实例,从而将客户端与TwoHole联系起来public TwoHole twoholeAdaptee = new TwoHole();/// <summary>/// 实现三个孔插头接口方法/// </summary>public override void Request(){twoholeAdaptee.SpecificRequest();}}
}

从上面代码可以看出,对象的适配器模式正如我们开始分析的思路去实现的, 其中客户端调用代码和类的适配器实现基本相同,下面让我们看看对象的适配器模式的类图,具体类图如下:

三、适配器模式的优缺点

  在引言部分已经提出,适配器模式用来解决现有对象与客户端期待接口不一致的问题,下面详细总结下适配器两种形式的优缺点。

类的适配器模式:

优点:

  1,可以在不修改原有代码的基础上来复用现有类,很好地符合 “开闭原则”
  2,可以重新定义Adaptee(被适配的类)的部分行为,因为在类适配器模式中,Adapter是Adaptee的子类
  3,仅仅引入一个对象,并不需要额外的字段来引用Adaptee实例(这个即是优点也是缺点)。
缺点:

  1,用一个具体的Adapter类对Adaptee和Target进行匹配,当如果想要匹配一个类以及所有它的子类时,类的适配器模式就不能胜任了。因为类的适配器模式中没有引入Adaptee的实例,光调用this.SpecificRequest方法并不能去调用它对应子类的SpecificRequest方法。
  2,采用了 “多继承”的实现方式,带来了不良的高耦合。

对象的适配器模式

优点:

  1,可以在不修改原有代码的基础上来复用现有类,很好地符合 “开闭原则”(这点是两种实现方式都具有的)
  2,采用 “对象组合”的方式,更符合松耦合。

缺点:

  1,使得重定义Adaptee的行为较困难,这就需要生成Adaptee的子类并且使得Adapter引用这个子类而不是引用Adaptee本身。

四、使用场景

在以下情况下可以考虑使用适配器模式:

  1,系统需要复用现有类,而该类的接口不符合系统的需求
  2,想要建立一个可重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。
  3,对于对象适配器模式,在设计里需要改变多个已有子类的接口,如果使用类的适配器模式,就要针对每一个子类做一个适配器,而这不太实际。

五、.NET中适配器模式的实现

1.适配器模式在.NET Framework中的一个最大的应用就是COM Interop。COM Interop就好像是COM和.NET之间的一座桥梁(关于COM互操作更多内容可以参考我的互操作系列)。COM组件对象与.NET类对象是完全不同的,但为了使.NET程序

象使用.NET对象一样使用COM组件,微软在处理方式上采用了Adapter模式,对COM对象进行包装,这个包装类就是RCW(Runtime Callable Wrapper)。RCW实际上是runtime生成的一个.NET类,它包装了COM组件的方法,并内部实现对COM组件的调用。如下图所示:

2..NET中的另外一个适配器模式的应用就是DataAdapter。ADO.NET为统一的数据访问提供了多个接口和基类,其中最重要的接口之一是IdataAdapter。DataAdpter起到了数据库到DataSet桥接器的作用,使应用程序的数据操作统一到DataSet上,而与具体的数据库类型无关。甚至可以针对特殊的数据源编制自己的DataAdpter,从而使我们的应用程序与这些特殊的数据源相兼容。

六、总结

  到这里适配器模式的介绍就结束了,本文主要介绍了适配器模式的两种实现、分析它们的优缺点以及使用场景的介绍,在适配器模式中,适配器可以是抽象类,并适配器模式的实现是非常灵活的,我们完全可以将Adapter模式中的“现存对象”作为新的接口方法参数,适配器类可以根据参数参数可以返回一个合适的实例给客户端。

以上内容摘抄自 http://learninghard.blog.51cto.com/6146675/1308323

转载于:https://www.cnblogs.com/xiongzaiqiren/p/5292629.html

设计模式(七)适配器模式(Adapter Pattern)相关推荐

  1. 乐在其中设计模式(C#) - 适配器模式(Adapter Pattern)

    [索引页] [源码下载] 乐在其中设计模式(C#) - 适配器模式(Adapter Pattern) 作者:webabcd 介绍 将一个类的接口转换成客户希望的另外一个接口.Adapter模式使得原本 ...

  2. 【设计模式】适配器模式 Adapter Pattern

    适配器模式在软件开发界使用及其广泛,在工业界,现实中也是屡见不鲜.比如手机充电器,笔记本充电器,广播接收器,电视接收器等等.都是适配器. 适配器主要作用是让本来不兼容的两个事物兼容和谐的一起工作.比如 ...

  3. 二十三种设计模式[6] - 适配器模式(Adapter Pattern)

    前言 适配器,属于类结构型模式.<设计模式 - 可复用的面向对象软件>一书中将之描述为" 将一个类的接口转换成客户希望的另一个接口,使得原本由于接口不兼容而不能一起工作的那些类可 ...

  4. 【设计模式】—— 适配器模式Adapter

    模式意图 如果已经有了一种类,而需要调用的接口却并不能通过这个类实现.因此,把这个现有的类,经过适配,转换成支持接口的类. 换句话说,就是把一种现有的接口编程另一种可用的接口. 模式结构 [类的适配器 ...

  5. 设计模式(七): 通过转接头来观察适配器模式(Adapter Pattern)

    在前面一篇博客中介绍了"命令模式"(Command Pattern),今天博客的主题是"适配器模式"(Adapter Pattern).适配器模式用处还是比较多 ...

  6. 极速理解设计模式系列:6.适配器模式(Adapter Pattern)

    四个角色:目标抽象类角色(Target).目标对象角色(Adapter).源对象角色(Adaptee).客户端角色(Client) 目标抽象类角色(Target):定义需要实现的目标接口 目标对象角色 ...

  7. 适配器模式 Adapter Pattern

    适配器模式 适配器模式(Adapter Pattern)是作为两个不兼容的接口之间的桥梁.这种类型的设计模式属于结构型模式,它结合了两个独立接口的功能. 这种模式涉及到一个单一的类,该类负责加入独立的 ...

  8. 『设计模式』适配器模式(Adapter)

    23种设计模式+额外常用设计模式汇总 (持续更新) 适配器模式(Adapter) 适配器模式把一一个类的接口变换成客户端所期待的另一种接口,从而使原本接口不匹配而无法在一起工作的两个类能够在一起工作. ...

  9. C++设计模式之适配器模式(adapter)(结构型)

    一.结构型模式概述 结构型模式(Structural Pattern)描述如何将类或者对象结合在一起形成更大的结构,就像搭积木,可以通过简单积木的组合形成复杂的.功能更为强大的结构. 结构型模式可以分 ...

  10. php本地服务手机适配器,php适配器模式(adapter pattern)

    下午陪家人和小孩,晚上练起来. /* The adapter pattern allows the interface of an existing class to be used from ano ...

最新文章

  1. springboot静态网页放哪里_STEP4.1 第一个SpringBoot项目
  2. kbengine定义实体
  3. dubbo kryo序列化_为什么如此高效?解密kryo各个数据类型的序列化编码机制,强...
  4. 教育部成立校外教育培训监管司 K12迎最强监管 教育中概股再跳水
  5. 多台kafka同时启动shell脚本
  6. 结构体赋值 -- 构造函数
  7. 视频教程-通俗易懂的JavaScript高级教程(含资料)-JavaScript
  8. 利用EasyPub为Kindle制作mobi格式书籍
  9. 数据结构--括号匹配检验(数据结构习题)
  10. Ubuntu 安装 tbb 步骤详解
  11. 同一局域网下,一台电脑连接另一台电脑的虚拟机(从属机(window)连接主机(window)虚拟机(Linux)的连接流程)
  12. 弘辽科技:淘宝流失率是什么意思?客户流失的原因有哪些?
  13. 内存类型UDIMM、RDIMM、LRDIMM
  14. mysql查询每个分组的最新数据
  15. 接口参数加解密,代码无侵入这样做方便多了
  16. css渐变效果的实现
  17. 一文看懂GFS如何搭建
  18. H - 互质数的个数(一)
  19. 被称为“2022大热门”的Android车载系统开发,到底应该怎么学?
  20. iperf软件编译以及使用

热门文章

  1. 为什么我要用C写游戏 适合自己的才是最好的
  2. SQL基础【十二、Like】
  3. Mac 解决brew一直卡在Updating Homebrew
  4. 同步一张表、复制过滤设置
  5. chrome 主进程cpu占用50%怎么办?
  6. vue全局路由守卫beforeEach
  7. face recognition[MobiFace]
  8. FJUT寒假第一周作业浮点数查寻题解
  9. Java集合框架总结(5)——Map接口的使用
  10. hdu 1159 Common Subsequence (dp)