深入浅出设计模式笔记之五:适配器模式

引子

昨天在给新买的MP3充电的时候,发现这款MP3播放器只提供了USB接口充电的方式,而它所配备的充电器无法直接给USB接口充电,聪明的厂商为充电器装上了一个USB接口转换器解决了问题。

这个USB接口转接器正是我们今天要谈到的适配器。而在软件开发中采用类似于上面方式的编码技巧被称为适配器模式。

 

二、定义和结构

《设计模式》一书中是这样给适配器模式定义的:将一个类的接口转换成客户希望的另外一个接口。Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。由引子中给出的例子可知,这个定义描述的功能和现实中的适配器的功能是一致的。

       可能你还是不太明白为什么要使用适配器模式。我们来举个例子也许能更直接的解除你的疑惑。

       比如,在一个画图的小程序中,你已经实现了绘制点、直线、方块等图形的功能。而且为了让客户程序在使用的时候不用去关心它们的不同,还使用了一个抽象类来规范这些图形的接口。现在你要来实现圆的绘制,这时你发现在系统其他的地方已经有了绘制圆的实现。在你庆幸之余,发现系统中已有的方法和你在抽象类中规定的方法名称不一样!这可怎么办?修改绘制圆的方法名,就要去修改所有使用它的地方;修改你的抽象类的方法名,也要去修改所有图形的实现方法以及已有的引用。还有其它的方法没有?那就是适配器模式了。

       可以看出使用适配器模式是为了在面向接口编程中更好的复用。如果你的系统中没有使用到面向接口编程,没有使用到多态,我想大概也不会使用到适配器模式。

       下面来看看适配器模式的组成吧。

1)         目标(Target)角色:定义Client使用的接口。

2)       被适配(Adaptee)角色:这个角色有一个已存在并使用了的接口,而这个接口是需要我们适配的。

3)       适配器(Adapter)角色:这个适配器模式的核心。它将被适配角色已有的接口转换为目标角色希望的接口。

放上一个简单的类图,这只是适配器模式实现的一种情况:

 

 

 

三、分类

       在《设计模式》一书中将适配器模式分为类适配器模式和对象适配器模式。区别仅在于适配器角色对于被适配角色的适配是通过继承完成的还是通过组合来完成的。由于在java中不支持多重继承,而且继承有破坏封装之嫌,众多的书中(包括《设计模式》)都提倡使用组合来代替继承。因此这里我们就不再对类适配器模式进行介绍(其实用的也很少)。

在上一小节的类图中描述的就是对象适配器模式。Adapter对Adaptee的转换是通过组合来完成的(如果你还搞不懂类图中基本元素的含义,请先阅读我的《UML类图介绍》)。

 

四、举例

       接着上面举的画图程序的例子,先来看看在添加绘制圆的需求前的类结构:

 

       添加了圆的绘制以后的类结构:

 

 

可以看出Shape、Circle和TextCircle三者的关系是和标准适配器模式中Target、Apater、Apatee三者的关系相对应的。我们只关心这个画图程序中是怎么来使用适配器模式的。看看Circle的实现代码吧:

 

class Circle extends Shape

{  

    //这里引用了TextCircle

private TextCircle tc;

public Circle ()

{    

tc= new TextCircle();       //初始化

}

void public display()

{

tc.displayIt();                   //在规定的方法里面调用TextCircle原来的方法

}

}

 

这样一个简单的适配器实现就完成了。

其实在适配器角色中不仅仅可以完成接口转换的过程,而且还可以对其功能进行改进和扩充,当然这就不属于适配器模式描述的范围内了。

前面我介绍过了代理模式,两者的主要区别在于代理模式应用的情况是不改变接口命名的,而且是对已有接口功能的一种控制;而适配器模式则强调接口转换。

 

 

五、题外话

在java中有一种叫做“缺省适配模式”的应用,它和我们所讲的适配器模式是完全的两种东西。缺省适配模式是为一个接口提供缺省的实现,这样子类型就可以从缺省适配模式中进行扩展,避免了从原有接口中扩展时要实现一些自己不关心的接口。在java.awt.event中的XXXAdapter就是它的很好的例子,有兴趣的可以看看。

 

六、总结

简单的介绍了下比较简单但是很实用的适配器模式。如有疏漏还请指正。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
适配器模式是一种结构设计模式,它允许将一个类的接口转换成客户端所期望的另一个接口。在Java中,可以使用以下步骤来实现适配器模式: 1. 定义目标接口(Target Interface):这是客户端所期望的接口。它描述了客户端可以使用的方法。 2. 创建适配器类(Adapter Class):适配器类实现了目标接口,并包含一个对适配者对象的引用。它将客户端的求转发给适配者。 3. 创建适配者类(Adaptee Class):适配者类是需要被适配的类,它包含了一些客户端不直接支持的方法。 下面是一个简单的示例,展示了如何在Java中实现适配器模式: ```java // 目标接口 interface Target { void request(); } // 适配者类 class Adaptee { public void specificRequest() { System.out.println("Adaptee: specificRequest"); } } // 适配器类 class Adapter implements Target { private Adaptee adaptee; public Adapter(Adaptee adaptee) { this.adaptee = adaptee; } public void request() { adaptee.specificRequest(); } } // 客户端代码 public class Main { public static void main(String[] args) { Adaptee adaptee = new Adaptee(); Target target = new Adapter(adaptee); target.request(); } } ``` 在上面的示例中,Adaptee类是需要被适配的类,它有一个specificRequest方法。Adapter类实现了Target接口,并将求转发给Adaptee类的specificRequest方法。客户端代码创建了一个Adapter对象,并通过调用Target接口的request方法来发起求。 这就是一个简单的Java实现适配器模式的例子。适配器模式可以帮助我们将不兼容的接口进行适配,提高代码的复用性和灵活性。希望能对您有所帮助!如果还有其他问题,随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值