01 定义
适配器模式,即将某个类的接口转换成客户端期望的另一个接口的表示,主要目的是实现兼容性,让原本因为接口不匹配,没办法一起工作的两个类,可以协同工作。
02 分类
- 类适配器
- 对象适配器
- 接口适配器
03 案例
需求
手机充电,通过手机充电器将220V电压适配为5V
方案一:类适配器
定义220V交流电(被适配者的角色)
/**
* 220V交流电(被适配者的角色)
* @author:liyajie
* @createTime:2022/2/17 21:41
* @version:1.0
*/
public class Ac {
public int outputAc(){
int srcV = 220;
System.out.println(srcV + "V交流电");
return srcV;
}
}
定义5v直流电(目标对象的角色)
/**
* 5v直流电(目标对象的角色)
* @author:liyajie
* @createTime:2022/2/17 21:44
* @version:1.0
*/
public interface Dc {
int outputDc();
}
定义适配器
/**
* 手机充电适配器
* @author:liyajie
* @createTime:2022/2/17 21:45
* @version:1.0
*/
public class PhoneAdapter extends Ac implements Dc{
@Override
public int outputDc() {
// 获取220V交流电
int srcV = outputAc();
// 模拟适配器过程,转换为5v直流电
int targetV = srcV / 44;
System.out.println("电压已经适配为" + targetV + "V" );
return targetV;
}
}
定义手机类
/**
* 手机类
* @author:liyajie
* @createTime:2022/2/17 21:48
* @version:1.0
*/
public class Phone {
public void charge(Dc dc){
if(dc.outputDc() == 5){
System.out.println("电压正常,可以安全充电");
}else {
System.out.println("电压异常,危险!");
}
}
}
定义测试类
/**
* 测试类
* @author:liyajie
* @createTime:2022/2/18 10:56
* @version:1.0
*/
public class Test {
public static void main(String[] args) {
new Phone().charge(new PhoneAdapter());
}
}
查看测试结果
方案二:对象适配器
该方案只需要改造手机适配器类即可,如下:
/**
* 手机充电适配器
* @author:liyajie
* @createTime:2022/2/17 21:45
* @version:1.0
*/
public class PhoneAdapter implements Dc {
private Ac ac;
public PhoneAdapter(Ac ac){
this.ac = ac;
}
@Override
public int outputDc() {
// 获取220V交流电
int srcV = ac.outputAc();
// 模拟适配器过程,转换为5v直流电
int targetV = srcV / 44;
System.out.println("电压已经适配为" + targetV + "V" );
return targetV;
}
}
改造测试类
/**
* 测试类
* @author:liyajie
* @createTime:2022/2/18 11:12
* @version:1.0
*/
public class Test {
public static void main(String[] args) {
new Phone().charge(new PhoneAdapter(new Ac()));
}
}
查看测试结果
方案三:接口适配器
需要改造的如下:定义一个默认的适配器,作用是实现Dc的多个方法,使其他的自定义适配器用来继承,扩展
/**
* 默认适配器
* @author:liyajie
* @createTime:2022/2/17 21:45
* @version:1.0
*/
public class DefaultAdapter implements Dc {
@Override
public int outputDc() {
return 0;
}
}
定义手机适配器
/**
* 手机充电适配器
* @author:liyajie
* @createTime:2022/2/17 21:45
* @version:1.0
*/
public class PhoneAdapter extends DefaultAdapter {
private Ac ac;
public PhoneAdapter(Ac ac){
this.ac = ac;
}
@Override
public int outputDc() {
// 获取220V交流电
int srcV = ac.outputAc();
// 模拟适配器过程,转换为5v直流电
int targetV = srcV / 44;
System.out.println("电压已经适配为" + targetV + "V" );
return targetV;
}
}
定义手机类
/**
* 手机类
* @author:liyajie
* @createTime:2022/2/17 21:48
* @version:1.0
*/
public class Phone {
public void charge(int v){
if(v == 5){
System.out.println("电压正常,可以安全充电");
}else {
System.out.println("电压异常,危险!");
}
}
}
定义测试类
/**
* 测试类
* @author:liyajie
* @createTime:2022/2/18 11:57
* @version:1.0
*/
public class Test {
public static void main(String[] args) {
new Phone().charge(new PhoneAdapter(new Ac()).outputDc());
}
}
查看测试结果
04 对比分析
方案一:类适配器
优势:由于它是继承被适配者,可以根据需求,重写被适配者的方法,灵活性较好
劣势:由于继承是单继承属性的,目标对象必须是一个接口,有一定的局限性
方案二:对象适配器
改造适配类,将继承被适配者,修改为持有被适配者,遵循合成复用原则,使用关联关系代替继承关系,进一步解耦,提高扩展性
方案三:接口适配器
接口适配器模式,又叫缺省适配器模式,通过设计一个中间抽象类,去实现接口,为接口中的每个方法都提供一个默认实现,然后再定义一个具体的适配器继承默认的适配器,只需要重写自己需要重写的方法即可
05 总结
适配器模式的三种实现方式,类适配器模式有一定的局限性,其中最常用的是对象适配器模式,所以比较推荐的是对象适配器模式
本文转载自微信公众号「温故知新之java」,可以通过以下二维码关注。转载本文请联系温故知新之java公众号。