适配器模式--java实现

285 阅读3分钟

使用插座的问题

现实生活中的适配器例子,比如:泰国的插座用的是两孔的(欧标),可以买个多功能转换插头(适配器),这样就可以使用了。

适配器模式基本介绍

  • 适配器模式(Adapter Pattern)将某个类的接口转换成客户端期望的另一个接口表示,主要的目的是兼容性,让原本因接口不匹配不能一起工作的两个类可以协同工作。其别名为包装器(Wrapper).
  • 适配器模式属于结构型模式
  • 主要分为三类:类适配器模式、对象适配器模式、接口适配器模式

适配器的工作原理

  • 适配器模式:将一个类的接口转换成另外一种接口,让原本接口不兼容的类可以兼容。
  • 从用户的角度看不到被被适配者,是解耦的。
  • 用户调用适配器转化出来的目标接口方法,适配器再调用被适配者的相关接口方法。
  • 用户收到反馈,感觉只是和目标接口交互,如图:

image.png

类适配器

基本介绍:Adapter类,通过继承src类,实现dest接口,完成src->dest的适配。

应用实例:以生活中充电器的例子来讲适配器,充电器本身相当于Adapter,220V交流点相当于src(即:被适配者),我们的目的是(dest:目标)5V直流电。

用类适配器解决手机充电的问题 image.png

代码

/**
 * src
 **/
public class Voltage220V {
    public int output220() {
        return 220;
    }
}
/**
 * dest
 **/
public interface IVoltage5V {
    int output();
}
/**
 * adapter
 **/
public class VoltageAdapter extends Voltage220V implements IVoltage5V {
    @Override
    public int output() {
        int voltage220V = this.output220();
        return voltage220V / 44;
    }
}
/**
 * client
 **/
public class Phone {
    public void charge(IVoltage5V voltage5V) {
        int voltage5VOutput = voltage5V.output();
        if (voltage5VOutput <= 5) {
            System.out.println("电压为5V,可以充电");
        } else {
            System.out.println("电压大于5V,不能充电");
        }
    }

    public static void main(String[] args) {
        Phone phone = new Phone();
        phone.charge(new VoltageAdapter());
    }
}

对象适配器

模式介绍

  • 基本思路是和类适配器模式相同,只是将Adapter类作修改,不是继承src类,而是持有src类的实例。即:持有src类,实现dest接口,完成src->dest的适配。
  • 对象适配器模式是适配器模式常用的一种。 对象适配器解决手机充电的问题

image.png

public class VoltageAdapter implements IVoltage5V {
    private Voltage220V voltage220V;

    public VoltageAdapter(Voltage220V voltage220V) {
        this.voltage220V = voltage220V;
    }

    @Override
    public int output() {
        int voltage220VOutPut = voltage220V.output220();
        return voltage220VOutPut / 44;
    }
}

注意事项及细节

  • 对象适配器和类适配器其实算是同一种思想,只不过实现方式不同。根据合成复用原则,使用组合替代继承,它解决了类适配器必须继承src的局限性问题,也不再要求dest必须是接口。
  • 使用成本更低、更灵活。

接口适配器

  • 一些书籍称为:适配器模式(Default Adapter Pattern)或缺省适配器。
  • 当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么抽象类的子类可有选择的覆盖父类的某些方法来实现需求。
  • 适用于一个接口不想使用其所有的方法的时候。

image.png

public interface Interface1 {
    void operation();

    void operation2();

    void operation3();

    void operation4();
}

public abstract class AbstractAdapter implements Interface1{
    @Override
    public void operation() {

    }

    @Override
    public void operation2() {

    }

    @Override
    public void operation3() {

    }

    @Override
    public void operation4() {

    }
}
public class Client {
    public static void main(String[] args) {
        new AbstractAdapter(){
            @Override
            public void operation() {
                System.out.println("operate1");
            }
        }.operation();
    }
}

适配器模式的注意事项及细节

  • 三种命名方式,是根据src是以怎样的形式给到Adapter(在Adapter里的形式)来命名的。
  1. 类适配器:以类给到,在Adapter里,就是当src当做类,继承。
  2. 对象适配器:以对象给到,在Adapter里,将src作为一个对象,持有。
  3. 接口适配器:以接口给到,在Adapter里,将src作为一个接口,实现。
  • Adapter模式最大的作用还是将原本不兼容的接口融合在一起工作。
  • 实际开发中,适配器模式不拘泥于上述三种经典形式。