实例讲解JAVA 适配器模式

在讲述这个模式之前,我们先看一个案例:中国球员去NBA打篮球

中国球员去NBA打篮球,可是他不懂英语,所以听不懂教练安排的战术,所以现在有三种解决方式

1、球员学会英语。2、教练学会中文。3、请个翻译。

1和2是长久之计,但不能解决迫在眉睫的问题。请个翻译是短暂的更好的选择。

放在软件设计层面上,这就叫做适配器模式。https://www.jb51.net/article/189484.htm

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

在软件开发中,也就是系统的数据和行为都正确,但接口不符时,我们应该考虑用适配器,目的是使控制范围之外的一个原有对象与某个接口匹配,适配器模式主要应用于希望复用一些现存的类,但是接口又与复用环境要求不一致的情况。

在 GoF 的设汁模式中,对适配器模式讲了两种类型,类适配器模式和对象适配器模式,由于类适配器模式通过多重继承对一个接口与另一个接口进行匹配,而JAVA语言不支持多重继承,也就是一个类只有一个父类,所以我们这里主要讲的是对象适配器。

Target(这是客户所期待的接口。目标可以是具体的或抽象的类,也可以是接口)代码如下:

public class Target {
 public void request() {
 System.out.println("普通请求");
 }
}

Adaptee(需要适配的类)代码如下:

public class Adaptee {
 public void specialRequest(){
 System.out.println("特殊请求");
 }
}

Adapter(通过在内部包装一个Adaptee对象,把原接口转换成目标接口)代码如下:

public class Adapter extends Target{
 private Adaptee adaptee = new Adaptee();//建立一个私有的Adaptee对象

 @Override
 public void request() {
 //这样就可以把表面上调用request()方法变成实际调用的specialRequest()
 adaptee.specialRequest();
 }
}

测试方法

public class Test {
 public static void main(String[] args) {
     //对main方法来说,调用的就是Target的request()
     Target target = new Target();
 target.request();
 }
}

看起来是不是很简单,是不是跟一句俗语很像“挂羊头卖狗肉”。

何时使用适配器模式?

在想使用一个已经存在的类,但如果它的接口,也就是它的方法和你的要求不相同时,就应该考虑用适配器模式。两个类所做的事情相同或相似,但是具有不同的接口时要使用它。而且由于类都共亨同一个接口,代码可以统一调用同一接口就行了,这样应该可以更简单、更直接、更紧凑。

其实用适配器模式也是无奈之举,很有点‘亡羊补牢'的感觉,没办法呀,是软件就有维护的一天,维护就有可能会因不同的开发人员、不同的产品、不同的厂家而造成功能类似而接口不同的情况,此时就是适配器模式大展拳脚的时候了。

现在把教练给球员们分配任务的例子用适配器模式实现

球员类

public abstract class Player {
 protected String name;
 public Player(String name) {
 this.name = name;
 }
 //进攻和防守方法
 public abstract void attack();
 public abstract void defense();
}

后卫、中锋、前锋类

//前锋
public class Forwards extends Player {
 public Forwards(String name) {
 super(name);
 }
 @Override
 public void attack() {
 System.out.println("前锋:"+name+"进攻");
 }
 @Override
 public void defense() {
 // TODO Auto-generated method stub
 System.out.println("前锋:"+name+"防守");
 }
}
//中锋
public class Center extends Player {
 public Center(String name) {
 super(name);
 }
 @Override
 public void attack() {
 System.out.println("中锋:"+name+"进攻");
 }

 @Override
 public void defense() {
 // TODO Auto-generated method stub
 System.out.println("中锋:"+name+"防守");
 }
}
//后卫
public class Guards extends Player {
 public Guards(String name) {
 super(name);
 }
 @Override
 public void attack() {
 System.out.println("后卫:"+name+"进攻");
 }
 @Override
 public void defense() {
 // TODO Auto-generated method stub
 System.out.println("后卫:"+name+"防守");
 }
}

测试方法

public class Test {
 public static void main(String[] args) {
 Player peter = new Forwards("peter");
 peter.attack();
 Player mike = new Guards("mike");
 mike.attack();

 Player zhangsan = new Center("张三");
 zhangsan.attack();
 zhangsan.defense();
 }
}

输出结果:

前锋:peter进攻
后卫:mike进攻
中锋:张三进攻
中锋:张三防守

球员“张三”不会说英语,需要翻译,用适配器模式完善代码

//外籍中锋
public class ForeignCenter {
 private String name;
 //外籍中锋只懂得中文“进攻”
 public void jingong() {
 System.out.println("中锋:"+name+"进攻");
 }
 //外籍中锋只懂得中文“防守”
 public void fangshou() {
 System.out.println("中锋:"+name+"防守");
 }

 //省略getter、setter方法
}
//翻译者
public class Translator extends Player {
 //声明并实例化一个内部“外籍中锋”对象,表面翻译者与外籍球员有关联
 private ForeignCenter wjzf = new ForeignCenter();

 public Translator(String name) {
 super(name);
 wjzf.setName(name);
 }

 @Override
 public void attack() {
 //翻译者将attack翻译为jingong 告诉外籍中锋
 wjzf.jingong();
 }

 @Override
 public void defense() {
 //翻译者将defense翻译为fangshou 告诉外籍中锋
 wjzf.fangshou();
 }
}

main方法修改代码如下:

//翻译者
public class Translator extends Player {
 //声明并实例化一个内部“外籍中锋”对象,表面翻译者与外籍球员有关联
 private ForeignCenter wjzf = new ForeignCenter();

 public Translator(String name) {
  super(name);
  wjzf.setName(name);
 }

 @Override
 public void attack() {
  //翻译者将attack翻译为jingong 告诉外籍中锋
  wjzf.jingong();
 }

 @Override
 public void defense() {
  //翻译者将defense翻译为fangshou 告诉外籍中锋
  wjzf.fangshou();
 }
}

现在就算是张三不懂英文,教练不懂中文,但因为有了翻译者,团队沟通合作成为了可能。

如果能事先预防接口不同的问题,不匹配问题就不会发生:在有小的接口不统一问题发生时,及时重构,问题不至于扩大:只有碰到无法改变原有设计和代码的情况时,才考虑适配。

事后控制不如事中控制,事中控制不如事前控制。

以上就是实例讲解JAVA 适配器模式的详细内容,更多关于JAVA 适配器模式的资料请关注我们其它相关文章!

(0)

相关推荐

  • java中设计模式之适配器模式

    java中设计模式之适配器模式  前言: 适配器模式可以将一个类或接口应用于另一个不同但是却有联系的接口,主要的做法是通过声明一个目标接口的实现类,在该类中声明一个将被适配类或接口(被适配者)作为参数的构造器和被适配者的实例,这样在实现目标接口的时候就可以调用被适配者的实例,并且辅以一些额外的操作.适配器模式的主体有三个部分:适配者,适配者实现类和被适配者.具体类结构如下图: 这里将被适配者的对象以组合的方式放到适配器类中,那么被适配者及其实现者都可以使用该适配器.适配器模式的优点在于可以将不同

  • 详解JAVA设计模式之适配器模式

    适配器模式 适配器模式(Adapter Pattern)是作为两个不兼容的接口之间的桥梁.这种类型的设计模式属于结构型模式,它结合了两个独立接口的功能. 这种模式涉及到一个单一的类,该类负责加入独立的或不兼容的接口功能.举个真实的例子,读卡器是作为内存卡和笔记本之间的适配器.您将内存卡插入读卡器,再将读卡器插入笔记本,这样就可以通过笔记本来读取内存卡. 我们通过下面的实例来演示适配器模式的使用.其中,音频播放器设备只能播放 mp3 文件,通过使用一个更高级的音频播放器来播放 vlc 和 mp4

  • 23种设计模式(5) java适配器模式

    23种设计模式第五篇:java适配器模式 定义: 将一个类的接口转换成客户希望的另外一个接口.适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作. 角色: 目标(Target)角色:这就是所期待得到的接口,也就是这类的接口是符合我们要求的. 源(Adapee)角色:我们要使用的接口,但是这个接口不符合我们的要求,也就是现在需要适配的接口. 适配器(Adaper)角色:适配器类是适配器模式的核心.适配器把源接口转换成目标接口.显然,这一角色不可以是接口,而必须是具体类. 分类: 1

  • Java设计模式之Adapter适配器模式

    一.场景描述 "仪器数据采集器"包含采集数据以及发送数据给服务器两行为,则可定义"仪器数据采集器"接口,定义两方法"采集数据capture"和"发送数据sendData". "PDF文件数据采集器"实现时,要实现"仪器数据采集器"接口,实现"采集数据"方法:目前有"PDF文件内容解析工具"类PdfFileExtractor,该类实现PDF文件的数据

  • 详解Java适配器模式

    假期刚结束不久,也没什么好写的,今天就水下文章,讲讲设计模式对开发时的影响,做开发到现在,感觉设计模式对开发的影响还是挺大的. 这次就简单谈谈适配器模式.可能适配器模式感觉比较鸡肋,但是用到的地方还挺多的,特别是合作开发的时候. 1.适配器模式 适配器模式,作为连接两个接口的桥梁.这个概念感觉有点那啥,很少用接口的朋友可能就没有什么感觉,经常面向接口编程的朋友比较能产生共鸣,简单来说,就是写一个适配器(转换器)来对接对象. 2.适配器模式使用 java适配器模式有两种,类适配器和对象适配器 (1

  • java设计模式笔记之适配器模式

    适配器(Adapter)模式: 适配器模式把一个类的接口变换成客户端所期待的另一种接口,从而使原本因接口不匹配而无法在一起工作的两个类能够在一起工作. 生活中的场景: 1.笔记本电源适配器,可以将220v转化为适合笔记本使用的电压. 2.给笔记本电脑的usb接口插入台式机的ps/2接口的键盘,需要一个usb和ps/2的接口转接器,此时usb和ps/2的接口转接器就充当了适配器的角色. 通用类图: 在上面的通用类图中,Cient 类最终面对的是 Target 接口(或抽象类),它只能够使用符合这一

  • 实例解析Java设计模式编程中的适配器模式使用

    平时我们会经常碰到这样的情况,有了两个现成的类,它们之间没有什么联系,但是我们现在既想用其中一个类的方法,同时也想用另外一个类的方法.有一个解决方法是,修改它们各自的接口,但是这是我们最不愿意看到的.这个时候Adapter模式就会派上用场了. Adapter模式也叫适配器模式,是由GoF提出的23种设计模式的一种.Adapter模式是构造型模式之一,通过Adapter模式,可以改变已有类(或外部类)的接口形式. 适配器 模式 有三种方式,一种是对象适配器,一种是类适配器, 一种是接口适配器 以下

  • Java适配器模式应用之电源适配器功能详解

    本文实例讲述了Java适配器模式应用之电源适配器功能.分享给大家供大家参考,具体如下: 一.模式定义 存在两种适配器模式 1 对象适配器模式,在这种适配器模式中,适配器容纳一个它包裹的类对象的物理实体. 2 类适配器模式,在这种适配器模式中,适配器继承自已实现的类. 二.模式举例 1 模式分析 我们借用笔计本电源适配器来说明这一模式. 已经存在的交流电源 笔记本电脑 电源适配器 2 适配器模式的静态建模 3 代码举例 3.1 抽象电源建立 package com.demo.power; /**

  • Java经典设计模式之适配器模式原理与用法详解

    本文实例讲述了Java经典设计模式之适配器模式.分享给大家供大家参考,具体如下: 适配器模式是把一个类的接口适配成用户所期待的,使得原本由于接口不兼容而不能一起工作的一些类可以在一起工作从而实现用户所期望的功能. 适配器模式的优势: 1. 通过适配器,客户端可以调用统一接口,操作简单直接,并且代码逻辑紧凑,使用起来方便. 2. 代码复用,适配器模式就是解决因为环境要求不相同 的问题,通过适配实现代码复用. 3. 将目标类和适配器类解耦,通过新建一个适配器类来重用现在的类,不用再去重复修改原有代码

  • 实例讲解JAVA 适配器模式

    在讲述这个模式之前,我们先看一个案例:中国球员去NBA打篮球 中国球员去NBA打篮球,可是他不懂英语,所以听不懂教练安排的战术,所以现在有三种解决方式 1.球员学会英语.2.教练学会中文.3.请个翻译. 1和2是长久之计,但不能解决迫在眉睫的问题.请个翻译是短暂的更好的选择. 放在软件设计层面上,这就叫做适配器模式.https://www.jb51.net/article/189484.htm 将一个类的接口转换成客户希望的另外一个接口.适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以

  • 实例讲解JAVA 模板方法模式

    在讲述这个模式之前,我们先看一个案例:抄题目:两个学生将老师出的题目抄写在纸上,并且写出答案 先看一个比较笨的写法 public class TestPaperA { public void testQuestion1(){ System.out.println("1+1等于几? a.1 b.2 c.3 d.4"); System.out.println("答案:b"); } public void testQuestion2(){ System.out.print

  • 实例讲解JAVA设计模式之备忘录模式

    在讲述这个模式之前,我们先看一个案例:游戏回档 游戏的某个场景,一游戏角色有生命力.攻击力.防御力等数据,在打Boss前和后会不一样,我们允许玩家如果感觉与Boss决斗的效果不理想,可以让游戏恢复到决斗前.下面是代码: 游戏角色类,用来存储角色的生命力.攻击力.防御力的数据. public class GameRole { private int vit;//生命力 private int atk;//攻击力 private int def;//防御力 //状态显示 public void st

  • 实例讲解Java HashSet

    HashSet 基于 HashMap 来实现的,是一个不允许有重复元素的集合. HashSet 允许有 null 值. HashSet 是无序的,即不会记录插入的顺序. HashSet 不是线程安全的, 如果多个线程尝试同时修改 HashSet,则最终结果是不确定的. 您必须在多线程访问时显式同步对 HashSet 的并发访问. HashSet 实现来 Set 接口. HashSet 中的元素实际上是对象,一些常见的基本类型可以使用它的包装类. 基本类型对应的包装类表如下: 基本类型 引用类型

  • 实例讲解Java中的synchronized

    一.使用场景 在负责后台开发的时候,很多时候都是提供接口给前端开发人员去调用,会遇到这样的场景: 需要提供一个领奖接口,每个用户名只能领取一次,我们可以将成功领取的用户在数据库用个标记保存起来.如果这个用户再来领取的时候,查询数据库看该用户是否领取过. 但是问题来了,假设用户手速很快,极短时间内点了两次领奖按钮(前端没有进行控制,我们也不能依赖前端去控制).那么可能掉了两次领奖接口,而且有可能第二次调用的时候查询数据库的时候,第一次领奖还没有执行完成更新领奖标记. 这种场景就可以使用到synch

  • 实例讲解Java 自旋锁

    一直以来不是怎么清楚自旋锁,最近有点时间,好好的学习了一下: 所谓的自旋锁在我的理解就是多个线程在尝试获取锁的时候,其中一个线程获取锁之后,其他的线程都处在一直尝试获取锁的状态,不会阻塞!!!那么什么叫做一直尝试获取锁呢?就是一个循环,比较经典的是AtomicInteger中的一个updateAndGet方法,下图所示(当然也可以直接看unsafe类中的getAndAddInt等类似方法): 我们可以看出在while循环中使用CAS去尝试更新一个变量,如果更新失败,就会一直在这个循环中一直在尝试

  • 实例讲解Java的Spring框架中的控制反转和依赖注入

    近来总是接触到 IoC(Inversion of Control,控制反转).DI(Dependency Injection,依赖注入)等编程原则或者模式,而这些是著名 Java 框架 Spring.Struts 等的核心所在.针对此查了 Wikipedia 中各个条目,并从图书馆借来相关书籍,阅读后有些理解,现结合书中的讲解以及自己的加工整理如下: eg1 问题描述: 开发一个能够按照不同要求生成Excel或 PDF 格式的报表的系统,例如日报表.月报表等等.   解决方案: 根据"面向接口编

  • 实例讲解Java设计模式编程中的OCP开闭原则

    定义:一个软件实体如类.模块和函数应该对扩展开放,对修改关闭. 问题由来:在软件的生命周期内,因为变化.升级和维护等原因需要对软件原有代码进行修改时,可能会给旧代码中引入错误,也可能会使我们不得不对整个功能进行重构,并且需要原有代码经过重新测试. 解决方案:当软件需要变化时,尽量通过扩展软件实体的行为来实现变化,而不是通过修改已有的代码来实现变化.          开闭原则是面向对象设计中最基础的设计原则,它指导我们如何建立稳定灵活的系统.开闭原则可能是设计模式六项原则中定义最模糊的一个了,它

  • 实例讲解Java编程中数组反射的使用方法

    什么是反射 "反射(Reflection)能够让运行于JVM中的程序检测和修改运行时的行为."这个概念常常会和内省(Introspection)混淆,以下是这两个术语在Wikipedia中的解释: 内省用于在运行时检测某个对象的类型和其包含的属性: 反射用于在运行时检测和修改某个对象的结构及其行为. 从它们的定义可以看出,内省是反射的一个子集.有些语言支持内省,但并不支持反射,如C++. 内省示例:instanceof 运算符用于检测某个对象是否属于特定的类. if (obj inst

  • 实例讲解Java的MyBatis框架对MySQL中数据的关联查询

    mybatis 提供了高级的关联查询功能,可以很方便地将数据库获取的结果集映射到定义的Java Bean 中.下面通过一个实例,来展示一下Mybatis对于常见的一对多和多对一关系复杂映射是怎样处理的. 设计一个简单的博客系统,一个用户可以开多个博客,在博客中可以发表文章,允许发表评论,可以为文章加标签.博客系统主要有以下几张表构成: Author表:作者信息表,记录作者的信息,用户名和密码,邮箱等. Blog表   :  博客表,一个作者可以开多个博客,即Author和Blog的关系是一对多.

随机推荐