Java设计模式透析之 —— 适配器(Adapter)

今天一大早,你的leader就匆匆忙忙跑过来找到你:“快,快,紧急任务!最近ChinaJoy马上就要开始了,老板要求提供一种直观的方式,可以查看到我们新上线的游戏中每个服的在线人数。”

你看了看日期,不是吧!这哪里是马上要开始了,分明是已经开始了!这怎么可能来得及呢?

“没关系的。”你的leader安慰你道:“功能其实很简单的,接口都已经提供好了,你只需要调用一下就行了。”

好吧,你勉为其难地接受了,对于这种突如其来的新需求,你早已习惯。

你的leader向你具体描述了一下需求,你们的游戏目前有三个服,一服已经开放一段时间了,二服和三服都是新开的服。设计的接口非常轻便,你只需要调用Utility.getOnlinePlayerCount(int),传入每个服对应的数值就可以获取到相应服在线玩家的数量了,如一服传入1,二服传入2,三服则传入3。如果你传入了一个不存在的服,则会返回-1。然后你只要将得到的数据拼装成XML就好,具体的显示功能由你的leader来完成。

好吧,听起来功能并不是很复杂,如果现在就开始动工好像还来得及,于是你马上敲起了代码。

首先定义一个用于统计在线人数的接口PlayerCount,代码如下:

[java] view
plain
copy

  1. public interface PlayerCount {
  2. String getServerName();
  3. int getPlayerCount();
  4. }

接着定义三个统计类实现了PlayerCount接口,分别对应了三个不同的服,如下所示:

[java] view
plain
copy

  1. public class ServerOne implements PlayerCount {
  2. @Override
  3. public String getServerName() {
  4. return "一服";
  5. }
  6. @Override
  7. public int getPlayerCount() {
  8. return Utility.getOnlinePlayerCount(1);
  9. }
  10. }

[java] view
plain
copy

  1. public class ServerTwo implements PlayerCount {
  2. @Override
  3. public String getServerName() {
  4. return "二服";
  5. }
  6. @Override
  7. public int getPlayerCount() {
  8. return Utility.getOnlinePlayerCount(2);
  9. }
  10. }

[java] view
plain
copy

  1. public class ServerThree implements PlayerCount {
  2. @Override
  3. public String getServerName() {
  4. return "三服";
  5. }
  6. @Override
  7. public int getPlayerCount() {
  8. return Utility.getOnlinePlayerCount(3);
  9. }
  10. }

然后定义一个XMLBuilder类,用于将各服的数据封装成XML格式,代码如下:

[java] view
plain
copy

  1. public class XMLBuilder {
  2. public static String buildXML(PlayerCount player) {
  3. StringBuilder builder = new StringBuilder();
  4. builder.append("<root>");
  5. builder.append("<server>").append(player.getServerName()).append("</server>");
  6. builder.append("<player_count").append(player.getPlayerCount()).append("</player_count>");
  7. builder.append("</root>");
  8. return builder.toString();
  9. }
  10. }

这样的话,所有代码就完工了,如果你想查看一服在线玩家数只需要调用:

[java] view
plain
copy

  1. XMLBuilder.buildXML(new ServerOne());

查看二服在线玩家数只需要调用:

[java] view
plain
copy

  1. XMLBuilder.buildXML(new ServerTwo());

查看三服在线玩家数只需要调用:

[java] view
plain
copy

  1. XMLBuilder.buildXML(new ServerThree());

咦?你发现查看一服在线玩家数的时候,返回值永远是-1,查看二服和三服都很正常。

你只好把你的leader叫了过来:“我感觉我写的代码没有问题,但是查询一服在线玩家数总是返回-1,为什么会这样呢?”

“哎呀!”你的leader猛然想起,“这是我的问题,前面没跟你解释清楚。由于我们的一服已经开放一段时间了,查询在线玩家数量的功能早就有了,使用的是ServerFirst这个类。当时写Utility.getOnlinePlayerCount()这个方法主要是为了针对新开的二服和三服,就没把一服的查询功能再重复做一遍。”

听到你的leader这么说,你顿时松了一口气:“那你修改一下Utility.getOnlinePlayerCount()就好了,应该没我什么事了吧?”

“晤。。。本来应该是这样的。。。可是,Utility和ServerFirst这两个类都已经被打到Jar包里了,没法修改啊。。。”你的leader有些为难。

“什么?这不是坑爹吗,难道要我把接口给改了?”你已经泪流满面了。

“这倒不用,这种情况下可以使用适配器模式,这个模式就是为了解决接口之间不兼容的问题而出现的。”

其实适配器模式的使用非常简单,核心思想就是只要能让两个互不兼容的接口能正常对接就行了。上面的代码中,XMLBuilder中使用PlayerCount这个接口来拼装XML,而ServerFirst并没有实现PlayerCount这个接口,这个时候就需要一个适配器类来为XMLBuilder和ServerFirst之间搭起一座桥梁,毫无疑问,ServerOne就将充当适配器类的角色。修改ServerOne的代码,如下所示:

[java] view
plain
copy

  1. public class ServerOne implements PlayerCount {
  2. private ServerFirst mServerFirst;
  3. public ServerOne() {
  4. mServerFirst = new ServerFirst();
  5. }
  6. @Override
  7. public String getServerName() {
  8. return "一服";
  9. }
  10. @Override
  11. public int getPlayerCount() {
  12. return mServerFirst.getOnlinePlayerCount();
  13. }
  14. }

这样通过ServerOne的适配,XMLBuilder和ServerFirst之间就成功完成对接了!使用的时候我们甚至无需知道有ServerFirst这个类,只需要正常创建ServerOne的实例就行了。

需要值得注意的一点是,适配器模式不并是那种会让架构变得更合理的模式,更多的时候它只是充当救火队员的角色,帮助解决由于前期架构设计不合理导致的接口不匹配的问题。更好的做法是在设计的时候就尽量把以后可能出现的情况多考虑一些,在这个问题上不要向你的leader学习。

适配器:将一个类的接口转换成客户希望的另外一个接口。适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。

时间: 2024-11-06 20:34:13

Java设计模式透析之 —— 适配器(Adapter)的相关文章

Java设计模式透析之 —— 单例(Singleton)

转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/8860649 写软件的时候经常需要用到打印日志功能,可以帮助你调试和定位问题,项目上线后还可以帮助你分析数据.但是Java原生带有的System.out.println()方法却很少在真正的项目开发中使用,甚至像findbugs等代码检查工具还会认为使用System.out.println()是一个bug. 为什么作为Java新手神器的System.out.println(),到了

Java设计模式透析之 —— 策略(Strategy)

今天你的leader兴致冲冲地找到你,希望你能够帮他一个小忙.他如今急着要去开会.要帮什么忙呢?你非常好奇. 他对你说.当前你们项目的数据库中有一张用户信息表.里面存放了非常用户的数据.如今须要完毕一个选择性查询用户信息的功能. 他说会传递给你一个包括很多username的数组.你须要依据这些username把他们对应的数据都给查出来. 这个功能非常easy的嘛.你爽快地答应了. 因为你们项目使用的是MySQL数据库,你非常快地写出了例如以下代码: [java] view plaincopy p

Java设计模式透析之 —— 模板方法(Template Method)

今天你还是像往常一样来上班,一如既往地開始了你的编程工作. 项目经理告诉你,今天想在server端添加一个新功能.希望写一个方法.能对Book对象进行处理.将Book对象的全部字段以XML格式进行包装.这样以后能够方便与client进行交互.而且在包装開始前和结束后要打印日志,这样方便调试和问题定位. 没问题!你认为这个功能简直是小菜一碟,很自信地開始写起代码. Book对象代码例如以下: [java] view plaincopy public class Book { private Str

Java设计模式透析之 —— 组合(Composite)

听说你们公司最近新推出了一款电子书阅读应用,市场反应很不错,应用里还有图书商城,用户可以在其中随意选购自己喜欢的书籍.你们公司也是对此项目高度重视,加大了投入力度,决定给此应用再增加点功能. 好吧,你也知道你是逃不过此劫了,没过多久你的leader就找到了你.他告诉你目前的应用对每本书的浏览量和销售量做了统计,但现在想增加对每个书籍分类的浏览量和销售量以及所有书籍总的浏览量和销售量做统计的功能,希望你可以来完成这项功能. 领导安排的工作当然是推脱不掉的,你只能硬着头皮上了,不过好在这个功能看起来

Java 平台透析

IEEE Spectrum 杂志发布了一年一度的编程语言排行榜,这也是他们发布的第四届编程语言 Top 榜. 据介绍,IEEE Spectrum 的排序是来自 10 个重要线上数据源的综合,例如 Stack Overflow.Twitter.Reddit.IEEE Xplore.GitHub.CareerBuilder 等,对 48 种语言进行排行. 与其他排行榜不同的是,IEEE Spectrum 可以让读者自己选择参数组合时的权重,得到不同的排序结果.考虑到典型的 Spectrum 读者需求

Headfirst设计模式的C++实现——适配器(Adapter)

duck.h 1 #ifndef _DUCK_H_ 2 #define _DUCK_H_ 3 4 class DUCK { 5 public: 6 virtual void quack() = 0; 7 virtual void fly() = 0; 8 }; 9 #endif mallard_duck.h 1 #ifndef _MALLARD_DUCK_H_ 2 #define _MALLARD_DUCK_H_ 3 4 #include "duck.h" 5 #include <

Java读源码学设计模式:适配器Adapter

适配器模式相关源码:slf4j-1.6.1.hibernate-3.6.7 大家都知道,log4j是一个广泛使用的日志工具,除此之外,sun公司在JDK中也有自己的日志工具,也就是java.util.logging.Logger.当然还有其他一些日志工具. 多种日志工具功能和使用方式类似,一般都包含debug.info.warn.error等日志级别的方法,但却没有实现共同的接口.这一点不像JDBC,虽然关系型数据库种类很多,例如MySQL.Oracle等,但是有一套统一的接口,也就是JDBC.

设计模式学习心得&lt;适配器 Adapter&gt;

适配器模式(Adapter Pattern)是作为两个不兼容的接口之间的桥梁.这种类型的设计模式属于结构型模式,它结合了两个独立接口的功能. 这种模式涉及到一个单一的类,该类负责加入独立的或不兼容的接口功能. 概述 意图 将一个类的接口转换成客户希望的另外一个接口. 适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作. 主要解决 主要解决在软件系统中,常常要将一些"现存的对象"放到新的环境中,而新环境要求的接口是现对象不能满足的. 何时使用 系统需要使用现有的类,而此类

Java设计模式之适配器模式(Adapter Pattern)

Adapter Pattern的作用是在不改变功能的前提下转换接口.Adapter分为两类,一类是Object Adapter, 另一类是Class Adapter.由于Class Adapter的实现需要用到多继承,而Java不支持多继承,所以这里只关注Object Adapter. 在JDK1.5之前是没有 java.util.Iterator 接口的,java.util.Enumeration 接口起着 Iterator 的作用.那么如果我们需要维护一些年代比较久远的代码,可能就会面临着没