适配器模式
适配器模式
转载自:https://blog.csdn.net/qq_45034708/article/details/113777338
定义
适配器模式(Adapter)的定义如下:将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类能一起工作。
使用场景
客户端需要一个target(目标)接口,但是不能直接重用已经存在的adapter(适配者)类,因为它的接口和target接口不一致,所以需要adapter(适配器)将adapter转换为target接口。前提是target接口和已存在的适配者adapter类所做的事情是相同或相似,只是接口不同且都不易修改。如果在设计之初,最好不要考虑这种设计模式。凡事都有例外,就是设计新系统的时候考虑使用第三方组件,因为我们就没必要为了迎合它修改自己的设计风格,可以尝试使用适配器模式。
类适配器
顾名思义,通过适配器通过类来实现,以类来继承和实现接口的方式,来获取被适配类的信息并转换输出重写到适配接口。即Adapter
类,通过继承src
类,实现dst
类接口,完成src
->dst
的适配。
// 被适配的类
class Voltage220V {
public int output220V() {
int src = 220;
System.out.println("电压=" + src);
return src;
}
}
// 被适配接口
interface IVoltage5V {
int output5V();
}
// 设配器类
class VoltageAdapter extends Voltage220V implements IVoltage5V {
@Override
public int output5V() {
int srcV = output220V();
// 转化为5V
int dstV = srcV / 44;
return dstV;
}
}
// 调用
class Phone {
public void charging(IVoltage5V iVoltage5V) {
if (iVoltage5V.output5V() == 5) {
System.out.println("电压是5v,可以充电");
} else if (iVoltage5V.output5V() > 5) {
System.out.println("电压大于5v,不可充电");
}
}
}
// 测试
public class ClassAdapter{
public static void main(String[] args) {
System.out.println("===适配器测试===");
Phone phone = new Phone();
phone.charging(new VoltageAdapter());
}
}
缺陷:
- 有一定局限性。因为类适配器需要继承
src类
,而Java是单继承机制,所以要求dst
必须是接口。 src
类的方法在Adapter
中都会暴露出来,耦合性高。- 可以根据需求重写
src
类的方法,使得Adapter
的灵活性增强了。
对象适配器
顾名思义,通过实例对象(构造器传递)来实现适配器,而不是再用继承,其余基本同类适配器。即:持有src
类,实现dst
类接口,完成src
->dst
的适配。
//被适配的类(不变)
class Voltage220V {
public int output220V() {
int src = 220;
System.out.println("电压=" + src);
return src;
}
}
//适配接口(不变)
interface IVoltage5V {
public int output5V();
}
// 适配器类
class VoltageAdapter implements IVoltage5V {
private Voltage220V voltage220V;
public VoltageAdapter(Voltage220V voltage220V) {
this.voltage220V = voltage220V;
}
@Override
public int output5V() {
int dst = 0;
if (null != voltage220V) {
int src = voltage220V.output220V();//获取220v电压
dst = src / 44;
}
return dst;
}
}
//调用(不变)
class Phone {
public void charging(IVoltage5V iVoltage5V) {
if (iVoltage5V.output5V() == 5) {
System.out.println("电压是5v,可以充电");
} else if (iVoltage5V.output5V() > 5) {
System.out.println("电压大于5v,不可充电");
}
}
}
public class ObjectAdapter {
public static void main(String[] args) {
System.out.println("===对象适配器");
Phone phone = new Phone();
VoltageAdapter voltageAdapter = new VoltageAdapter(new Voltage220V());
phone.charging(voltageAdapter);
}
}
优缺点:
- 把继承解耦,解决了类适配器必须继承
src
的局限性问题,要求dst
必须是接口。 - 成本低更灵活。
接口适配器
继承那边可以解耦了,那能不能从接口这边解耦?
接口适配器也称缺省适配器模式,适用于一个接口不想使用其所有的方法的情况。当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆盖父类的某些方法来实现需求。
// 被适配的类(不变)
class Voltage220V {
public int output220V() {
int src = 220;
System.out.println("电压=" + src);
return src;
}
}
// 适配接口
interface IVoltage5V {
int output5V();
void m2(); //接口中不重要的方法
}
// 抽象适配器
abstract class AbsAdapter extends Voltage220V implements IVoltage5V {
// 以空方法实现接口所有方法
@Override
public int output5V() {
return 0;
}
@Override
public void m2() {
}
}
// 调用(不变)
class Phone {
public void charging(IVoltage5V iVoltage5V) {
if (iVoltage5V.output5V() == 5) {
System.out.println("电压是5v,可以充电");
} else if (iVoltage5V.output5V() > 5) {
System.out.println("电压大于5v,不可充电");
}
}
}
// 测试
public class InterfaceAdapter {
public static void main(String[] args) {
System.out.println("===接口适配器");
AbsAdapter absAdapter = new AbsAdapter() { //匿名内部类形式
@Override //按需要重写接口方法
public int output5V() {
System.out.println("使用了output5V的方法");
int srcV = output220V();
int dstV = srcV / 44; //转成5v
return dstV;
}
};
Phone phone = new Phone();
phone.charging(absAdapter);
}
}
总结
以上三种形式是根据src
是以怎样的形式给到Adapter来命名的:
- 类适配器:以类给到,在Adapter里,就是将
src
当做类,继承。 - 对象适配器:以对象给到,在Adapter里,将
src
作为一个对象,持有。 - 接口适配器:以接口给到,在Adapter里,将
src
作为一个接口,实现。
Adapter模式最大的作用还是将原本不兼容的接口融合在一起工作,在实际开发中,实现起来不拘泥于我们讲解的三种经典形式。