设计模式学习5 -- 适配器模式(Adapter)

 

1.   适配器模式(Adapter

1.1. 单向适配器模式

1.1.1.  概述

l         场景:有两个类

n         Class A{ public void funA(){/*A代码 */ }}

n         Class B{ public void funB(){/*B代码 */ }}

l         这两个模块本来没有任何关系,但是应用中需要凡是以后A的对象调用了funA方法,一定要同时调用BfunB方法,怎么办?不能改变源代码

l         这个问题的类比:贵公司开发了一套图像应用软件,其中一个功能是可以把图像进行清晰化。但是这套软件运行一段时间之后,大家发现将图片清晰化之后然后进行一下去噪声可能会更好。于是贵公司又开发了一个专门去噪声的类,现在问题来了,怎么样把这个去噪声的类接到你原来的软件上?不改变源代码。

1.1.2.  解决办法:使用单向适配器模式

适配器模式分为单向适配器模式,如果将B接到A上用,这叫作单向适配器模式,A主动,B被动。单向适配器模式要点:继承方法。

代码如下:

class A{

        public void funA(){

             System.out.println("图像清晰化");

             }

        }

class B{

       public void funB(){

              System.out.println("图像去噪声老算法");

              }

       }

class SubA extends A{//A进行扩展,以后将SubA当成A来用

       public void funA(){

              super.funA();//值得一提的是,这句代码可以去掉,如果去掉,就是功能替换

              B b=new B();

b. funB();

              }

       }

 

public class Adapter1{//客户端,最终使用方

       public static void main(String args[]){

              SubA subA = new SubA();

              subA.funA();

              }

       }

在上面代码中,SubA 中继承了A,这样SubA就拥有A的所有功能,重写funA方法,把B对象的funB方法加进来,但在生产B对象时,直接使用new方法,耦合性太强,最好面对接口编程,是B对象的生成是由客户但传入进来的。修改代码如下:

class A{

        public void funA(){

             System.out.println("图像清晰化");

             }

        }

        

interface IEM{//定义一个接口,管理所有的扩展模块(被动方模块)

       public abstract void extendFun();

       }

      

class B implements IEM{

       public void extendFun(){

              System.out.println("图像去噪声老算法");

              }

       }

      

class C implements IEM{

       public void extendFun(){

              System.out.println("图像去噪声新算法");

              }

       }    

 

class SubA extends A{//A进行扩展,以后将SubA当成A来用

       private IEM iem;    //完全面向接口编程,可以将IEM的任何实现类对象接入(可以用Spring管理)

       public SubA(IEM iem){

              this.iem = iem;

              }

       public void funA(){

              super.funA();//值得一提的是,这句代码可以去掉,如果去掉,就是功能替换

              iem.extendFun();

              }

       }

 

public class Adapter1{//客户端,最终使用方

       public static void main(String args[]){

              C c = new C();

              SubA subA = new SubA(c);

              subA.funA();

              }

       }

 

1.1.3.  单向适配器要点

l         注意:分清主动和被动方,主动方称为Adaptor,被动方称为Adaptee

l         定义一个类SubAdaptor,继承Adaptor类,重写funA方法

l         在这个类的构造函数可以传入Adaptee的引用

l         SubAdaptor当成Adaptor来使用

l         Adaptor a = new SubAdaptor(new Adaptee());

l         一种更好的方法:用接口,spring管理Adaptee

1.2. 双向适配器

1.2.1.  概述

l         代码:

Class A{

public void funA1(){ /* A1代码 */ }

public void funA2(){/* A2代码 */}

}

Class B{

public void funB1(){ /* B1代码 */ }

public void funB2(){/* B2代码 */}

}

 

l         应用需要凡是以后A的对象调用了funA1方法,一定要同时调用BfunB1方法;B的对象调用了funB2方法,一定要同时调用AfunA2方法;怎么办?不能改变源代码

1.2.2.  解决办法:使用双向适配器模式

双向适配模式是使用组合法将两个对象的方法进行组合。例如:

class A {

       public void funA1(){

              System.out.println("A.funA1");

              }

       public void funA2(){

              System.out.println("A.funA2");

              }

}

class B {

       public void funB1(){

              System.out.println("B.funB1");

              }

       public void funB2(){

              System.out.println("B.funB2");

              }

}

 

class AB {//组合AB,AB当成AB来使用

       private A a;

       private B b;

       public void setA(A a){

              this.a = a;

              }

       public void setB(B b){

              this.b = b;

              }

       ///功能编写

       public void funA1(){

              a.funA1();//这句话如果去掉,就是功能替换

              b.funB1();

              }

       public void funB2(){

              b.funB2();

              a.funA2();

              }    

       }

 

 

public class Adapter2{//客户端,最终使用方

       public static void main(String args[]){

              AB ab = new AB();

              ab.setA(new A());

              ab.setB(new B());

              ab.funA1();

              ab.funB2();

              }

       }

以上代码没有使用面向接口编写,修改如下:

interface IA{

       public abstract void funA1();

       public abstract void funA2();

       }

class A implements IA{

       public void funA1(){

              System.out.println("A.funA1");

              }

       public void funA2(){

              System.out.println("A.funA2");

              }

}

 

interface IB{

       public abstract void funB1();

       public abstract void funB2();

       }

class B implements IB{

       public void funB1(){

              System.out.println("B.funB1");

              }

       public void funB2(){

              System.out.println("B.funB2");

              }

}

 

class AB implements IA,IB{//组合AB,AB当成AB来使用

       private IA ia;

       private IB ib;

       public void setIa(IA ia){//可以通过Spring框架来进行装配

              this.ia = ia;

              }

       public void setIb(IB ib){

              this.ib = ib;

              }

       ///功能编写

       public void funA1(){

              ia.funA1();//这句话如果去掉,就是功能替换

              ib.funB1();

              }

       public void funB2(){

              ib.funB2();

              ia.funA2();

              }    

       //重写其他功能,以免丢失

       public void funA2(){

              ia.funA2();

              }

       public void funB1(){

              ib.funB1();

              }

       }

 

 

public class Adapter2{//客户端,最终使用方

       public static void main(String args[]){

              AB ab = new AB();

              ab.setIa(new A());

              ab.setIb(new B());

              ab.funA1();

              ab.funB2();

              }

       }

1.2.3.  双向适配器模式要点

l         因为不允许多重继承,用接口将两个类以及相关方法管理起来

l         编写第三个类,在构造函数传入前两个类的引用,并实现接口,重写函数

l         在函数里面进行调用

l         第三个类既可以当A,又可以当B来使用

1.2.4.  双向适配器练习

l         练习:有一个Frame1,上有一个按钮,点击按钮能让Frame1变红;另外有一个Frame2上有一个按钮,点击能将其标题改为China

l         现在要求:这两个Frame在开发的时候并不知道将要调用外界的类,将这两个类合起来,达到:点击Frame1内的按钮,出现Frame2,能将Frame2的标题改为China,点击Frame2内的按钮,将Frame1变红。不改变Frame12源代码

1.3. 适配器小结

把一个类的接口变换成客户端所期待的另一种接口,从而是原本因接口原因不匹配而无法一起工作的两个类能够一起工作。适配器可以根据参数返还一个合适的实例给客户端。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值