zoukankan      html  css  js  c++  java
  • 适配器模式--Adapter Pattern

      模拟场景:很多人都喜欢看NBA吧,姚明进驻NBA,打开了中国的市场。虽然后面姚明在NBA打得还不错,但是在刚进入NBA篮坛的时候,并不是那么顺利的。语言交流就是一个最大的问题。刚开始打球期间,教练及队员的战术部署姚明都无法理解,所以他需要这么一个翻译者,将教练及队员的意思转达给姚明,这样才可以进行合作。

      现在进行场景的模拟,先不考虑那么多。假如姚明一开始进入NBA的时候就已经会英语,可以进行交流了。那么这个时候教练就可以进行战术的部署了。

      转换成类,所有的队员都要服从教练的战术要求,假设现在教练都要给队员部署进攻和防守的指令。所有大家接收的动作是一致的,只是实现不一样而已。所有这里需要先定义出一个接口或者抽象类。

    IPlayer 

     1 package com.zqz.dp.adapter;
     2 /**
     3  * @author Qin 所有运动员的父接口,完成同样的动作
     4  */
     5 public interface IPlayer {
     6     /**
     7      * 运动员进攻
     8      */
     9     void attack();
    10     /**
    11      * 运动员防守
    12      */
    13     void defense();
    14 }

      构建完接口之后,就是对接口进行实现了。假设现在有前锋forwards、中锋center、后卫guards三种运动员,来根据教练的指令完成不同的动作。

    Forwards

     1 package com.zqz.dp.adapter;
     2 /**
     3  * @author Qin
     4  * 前锋类,实现运动员接口,实现自己的动作
     5  */
     6 public class Forwards implements IPlayer {
     7     private String name;    //定义运动员的名字
     8     public Forwards(String name) {        //构造方法赋值
     9         this.name = name;
    10     }
    11     @Override
    12     public void attack() {
    13         System.out.println("前锋〖"+this.name+"〗攻击。。。");
    14     }
    15     @Override
    16     public void defense() {
    17         System.out.println("前锋〖"+this.name+"〗防守。。。");
    18     }
    19 }

    Center

     1 package com.zqz.dp.adapter;
     2 /**
     3  * @author Qin
     4  * 中锋类,实现运动员接口,实现自己的动作
     5  */
     6 public class Center implements IPlayer {
     7     private String name;    //定义运动员的名字
     8     public Center(String name) {        //构造方法赋值
     9         this.name = name;
    10     }
    11     @Override
    12     public void attack() {
    13         System.out.println("中锋〖"+this.name+"〗攻击。。。");
    14     }
    15     @Override
    16     public void defense() {
    17         System.out.println("中锋〖"+this.name+"〗防守。。。");
    18     }
    19 }

    Guards

     1 package com.zqz.dp.adapter;
     2 /**
     3  * @author Qin
     4  * 后卫类,实现运动员接口,实现自己的动作
     5  */
     6 public class Guards implements IPlayer {
     7     private String name;    //定义运动员的名字
     8     public Guards(String name) {        //构造方法赋值
     9         this.name = name;
    10     }
    11     @Override
    12     public void attack() {
    13         System.out.println("后卫〖"+this.name+"〗攻击。。。");
    14     }
    15     @Override
    16     public void defense() {
    17         System.out.println("后卫〖"+this.name+"〗防守。。。");
    18     }
    19 }

    Client

     1 package com.zqz.dp.adapter;
     2 /**
     3  * @author 
     4  * Qin 场景类,模拟NBA比赛
     5  */
     6 public class Client {
     7     public static void main(String[] args) {
     8         IPlayer ba=new Forwards("巴蒂尔");
     9         ba.attack();
    10         IPlayer yao=new Center("姚明");
    11         yao.attack();
    12         yao.defense();
    13         IPlayer mai=new Guards("麦迪");
    14         mai.defense();
    15     }
    16 }

      以上的操作很简单,说白了就是父类引用指向子类实例,但是现在姚明是直接可以听懂了教练的战术安排了,这个已经是后面发展得很好的姚明了。并不是一开始不能进行交流的那个大姚了。

      要模拟刚入NBA的大姚,那么说明需要创建一个新的类来表示大姚这个对象。在这里我们把Center改成foreignCenter,表示外籍中锋。而很明显这个时候大姚根本无法听懂教练的意思,所以需要一个翻译者translator,翻译者是可以听懂教练的意思的,但是听懂了总不能自己上场比赛吧,所以要把教练的意思传递给大姚,交给大姚去执行。

      转化为类之间的关系也就是translator类要实现IPlayer接口,并且要有一个fireignCenter的引用,具体的实现交给引用去调用自己的方法。

    ForeignCenter

     1 package com.zqz.dp.adapter;
     2 /**
     3  * @author Qin 
     4  * 外籍中锋类,外籍中锋是新引进的类,并不能听懂教练的意思,所以只能执行自己的功能
     5  */
     6 public class ForeignCenter {
     7     private String name; // 定义运动员的名字
     8     public ForeignCenter(String name) { // 构造方法赋值
     9         this.name = name;
    10     }
    11     public void attack() {
    12         System.out.println("外籍中锋〖" + this.name + "〗攻击。。。");
    13     }
    14     public void defense() {
    15         System.out.println("外籍中锋〖" + this.name + "〗防守。。。");
    16     }
    17 }

    Translator

     1 package com.zqz.dp.adapter;
     2 /**
     3  * @author Qin
     4  * 翻译者类,适配器对象。传递教练的意思给外籍中锋
     5  */
     6 public class Translator implements IPlayer {
     7     private ForeignCenter fc;
     8     public Translator() {    //翻译者类知道要把消息传递给谁,所以立即生成外籍中锋实例
     9         fc=new ForeignCenter("大姚");
    10     }
    11     @Override
    12     public void attack() {
    13         this.fc.attack();    //告诉大姚要进行攻击
    14     }
    15     @Override
    16     public void defense() {
    17         this.fc.defense();    //告诉大姚进行防守
    18     }
    19 }

      其他类不变。这样就完成了功能,由翻译者转达大姚进行战术的安排。

      上面的操作其实很简单,只是在执行的时候调用的是具体真实类的操作而已。是否会觉得有点跟代理类似,其实会这样觉得是因为我把ForeignCenter中的方法定义成和IPlayer的方法一致。也就是说虽然教练给大姚下达了战术的安排,但是大姚可以执意按自己的想法进行安排。所以这也是和代理不同的地方。

      以上的操作就是适配器模式,准确来说是对象适配器,因为在类中传递了另一个类的引用。还有另一种叫做类适配器。要实现类适配器就方便多了,只要让Translator继承ForeignCenter即可,再调用父类的方法。很明显在这里是不合适的,因为两者不存在is-a的关系,所以在真正开发中,对象适配器用的必类适配器多。

      听名字大家就应该能明白意思了,何为适配器?就是把两个不合适的对象通过一个中间对象连接起来。像我们平常的电池充电不也需要一个电源适配器吗?就是一样的道理。

      适配器模式是一个补救的模式,也就是说一般在开发前期不会考虑该模式,而是在后期扩展中才会考虑到。还是以上面的例子为例,NBA事先根本就不能料想到是否会有中国球员参加比赛,所以才会有翻译者的存在,翻译者就是一个适配器。

    适配模式的优点:

      1、  适配器可以将两个毫无关联的对象连接起来。

      2、  增加了类的透明度:也就是说教练根本不用管大姚,直接告诉翻译者如何做就行了。

      3、  提高了类的复用性,易扩展,灵活性好。

    适配器的使用场景:

      你有动机去修改一个已经投入生产的接口时,就可以使用适配器进行扩展。

    适配器的拓展:

      除了类适配器和对象适配器外,还有一种缺省适配器,所谓的缺省适配器就是说在一个接口中存在了一大堆的方法,但是我们在具体的实现中根本就不需要这么多方法,那么我们就可以创建一个中间类,把这些方法全部空实现,然后具体的类再继承该空实现的类,完成自己所需要的功能。

  • 相关阅读:
    tcp/ip ---IP路由选择
    tcp/ip --IP:网际协议
    tcp/ip ---以太网和IEEE 802封装
    tcp/ip ---链路层
    internet与Internet的区别
    TCP/IP ---封装与分用
    TCP/IP ---互联网的地址
    SpringMvc 系统启动时加载数据到内存中
    hadoop-17-hive数据库元数据查看
    hadoop-16-sqoop导入oracle数据
  • 原文地址:https://www.cnblogs.com/littleQin/p/3695482.html
Copyright © 2011-2022 走看看