适配器模式将某个类的接口转换成客户端期望的另一个接口表示,主要目的是兼容性,让原本因接口不匹配不能一起工作的两个类可以协同工作。
适配器模式属于结构性模式,主要分为三类:类适配器模式、对象适配器模式、接口适配器模式。
什么是类适配器模式?
Adapter类,通过继承src类,实现dst类接口,完成src—>dst的适配。
应用实例:
以生活中充电器的例子来讲解适配器,充电器本身相当于Adapter,220V交流电相当于src(即被适配者),我们的dst(即目标)是5V直流电。
被适配者:
public class Voltage220V {//输出220V电压public int output220V(){int src=220;System.out.println("电压="+src+"伏");return src;}}
目标:
public interface IVoltage5V {public int output5V();}
适配器:
public class VoltageAdapter extends Voltage220V implements IVoltage5V{@Overridepublic int output5V() {//获取220V电压int srcV = output220V();int destV=srcV/44;return destV;}}
通过继承 被适配者 —->拥有该被适配者的功能,然后进一步修改成想要的功能 并 返回给目标
客户端:
public class Client {public static void main(String[] args) {Phone phone = new Phone(); //需要5V电压phone.charging(new VoltageAdapter());}}
类适配器模式注意事项和细节:
Java是单继承机制,所以类适配器需要继承src类这一点算是一个缺点,因为这要求dst必须是接口,有一定局限性。
src类的方法在Adapter中都会暴露出来,也增加了使用的成本(extends)。
由于其继承了src类,所以它可以根据需求重写src类的方法,使得Adapter的灵活性增强了。
对象适配器模式:
被适配者:
public class Android {public void isAndroid(){System.out.println("这是一个只接受安卓充电线的插口");}}
目标:
public class Iphone {public void isIphone(){System.out.println("这是一个适配苹果充电线的插口");}}
适配器:
/*** 适配器,作为中间件,把他们进行适配*/public class Adapter extends Android{private Iphone iphone;public Adapter(Iphone iphone){this.iphone = iphone;}@Overridepublic void isAndroid() {iphone.isIphone();}}
对象适配器和类适配器其实算是同一种思想,只不过实现方式不同。根据合成复用原则,使用组合替代继承,将被适配目标通过构造器传进来—->拥有被适配者的功能,然后进一步改造成想要的功能。
它解决了类适配器必须继承src的局限性问题,也不再要求dst必须是接口。
使用成本更低更灵活。
客户端:
public class Demo {public static void main(String[] args){Android android = new Adapter(new Iphone());//调用的是安卓的接口,但实际上通过适配器改造成一个可以接受苹果充电器的接口android.isAndroid();}}
接口适配器模式:
当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆盖父类的某些方法来实现需求。
适用于一个接口不想使用其所有的方法的情况。
接口:
public interface Interface4 {public void m1();public void m2();public void m3();public void m4();}
抽象类实现接口:
public abstract class AbsAdapter implements Interface4 {@Overridepublic void m1() {}@Overridepublic void m2() {}@Overridepublic void m3() {}@Overridepublic void m4() {}}
客户端:
public class Client {public static void main(String[] args) {AbsAdapter absAdapter =new AbsAdapter(){//只需要去覆盖需要使用的方法@Overridepublic void m1() {System.out.println("使用m1方法");}};absAdapter.m1();}}
