1. 适配器模式概述
① 什么是适配器模式?
- 适配器模式可以将一个类的接口转化为客户希望的另一种接口,使得因为接口不兼容而无法一起工作的类可以一起工作。
- 自己的理解:
- 某个应用的原始版本定义的请求方法叫
onRequest()
,现在新的版本中需要统一使用request()
进行命名。 - 这时可以定义一个目标接口,该接口中有抽象方法
request()
。 - 创建适配器类去实现目标接口,适配器类中将旧版本对象(待适配对象)作为内部对象。
- 在
request()
方法中调用待适配对象的onRequest()
方法,这样在使用者看来就像是在使用新方法一样,不会出现由于方法名变化而不可使用的情况。
② 适配器模式举例
- 读卡器和手机内存卡: 你想通过查看相机内存卡上的照片、视频,电脑上没有直接的插槽,你需要借助读卡器。这时,读卡器是Adapter,内存卡是Adaptee,电脑是目标接口。
- Lightning和圆孔耳机: iPhone 7上取消了耳机插孔,想听音乐只能通过充电插口。这时通过Lightning可以将圆孔耳机连接到充电插口,因为Lightning一端是圆孔耳机插口,一端是充电插头。这时Lightning是Adapter,圆孔耳机是Adaptee,iPhone 7是目标接口。
③ 适配器模式中的几种角色
- 目标接口: 定义了Adapter(适配器)的执行接口。
- Adaptee: 待适配类,里面的执行接口不符合使用要求,需要由Adapter进行转化。
- Adapter: 实现目标接口,通过对象适配器方式或类适配器方式,实现对Adaptee中某些方法的转化,使其满足使用要求。
2. 对象适配器
① 对象适配器的类图
- 对象适配器的主要思想:
将待适配对象作为适配器类的内部对象,在要求的执行接口中调用待适配对象不符合要求的方法,实现对不符合用户要求的接口的转化。
![在这里插入图片描述](https://img-blog.csdnimg.cn/20190912105154916.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3UwMTQ0NTQ1Mzg=,size_16,color_FFFFFF,t_70)
② 对象适配器编程实例
- 创建目标接口,目标接口中包含符合要求的执行接口。
- 创建待适配器类,包含不兼容的执行接口。
- 创建适配器类,实现目标接口并将待适配器对象作为内部对象,在执行接口中调用待适配器对象的执行接口。
interface Target {
void request();
}
class Adaptee {
public void specificRequest() {
System.out.println("我是旧版本的请求处理方法");
}
}
class Adapter implements Target {
private Adaptee adaptee;
public Adapter(Adaptee adaptee) {
this.adaptee = adaptee;
}
@Override
public void request() {
System.out.println("我是适配器,你现在可以使用旧版本的处理方法了");
adaptee.specificRequest();
}
}
public class ObjectAdapter {
public static void main(String[] args) {
Adaptee adaptee = new Adaptee();
Adapter adapter = new Adapter(adaptee);
adapter.request();
}
}
- 代码运行截图:
![在这里插入图片描述](https://img-blog.csdnimg.cn/20190912144436316.png)
3. 类适配器
① 类适配器的类图
- 类适配器的主要思想:
适配器类继承待适配器类并实现目标接口,在目标执行接口中直接调用父类(待适配器类)的执行接口,便可以将待适配器类中不兼容的执行接口转化为要求的执行接口。
![在这里插入图片描述](https://img-blog.csdnimg.cn/20190912161158784.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3UwMTQ0NTQ1Mzg=,size_16,color_FFFFFF,t_70)
② 类适配器编程实例
- 创建目标接口,目标接口中包含符合要求的执行接口。
- 创建待适配器类,待适配器类中包含不符合要求的执行接口。
- 创建适配器类,继承待适配器类并实现目标接口,直接在符合要求的执行接口中调用父类的执行接口。
interface Target {
void request();
}
class Adaptee {
public void specificRequest() {
System.out.println("我是旧版本的请求处理方法");
}
}
class Adapter1 extends Adaptee implements Target {
@Override
public void request() {
System.out.println("我是适配器,你现在可以使用旧版本的处理方法了");
super.specificRequest();
}
}
public class ClassAdapter {
public static void main(String[] args) {
Adapter1 adapter1 = new Adapter1();
adapter1.request();
}
}
4. 适配器总结
① 对象适配器和类适配器的区别
- 采用的方式不同: 类适配器使用继承方式,是静态的定义方式;对象适配器使用对象组合方式,是动态组合的定义方式。
- 服务范围的限制: 类适配器使用继承方式,使得Adapter只能服务于被继承的Adaptee;对象适配器使用对象这方式,可以同时服务于被继承的Adaptee及其子类。
- 重定义Adaptee的行为: 类适配器可以很方便的重定义Adaptee的行为,因为子类可以重写父类;对象适配器重定义Adaptee的行为则比较困难:如果想要重定义Adaptee的行为,需要创建Adaptee的子类并与子类组合。
- 额外的引用: 类适配器只需要创建一个Adapter对象,并不需要额外的Adaptee对象;对象适配器需要创建额外的Adaptee对象来间接引用Adaptee。
- 使用选择:
类适配器具有高耦合,灵活性低的特点;对象适配器则低耦合,灵活性高,一般选择使用对象适配器。
② 适配器模式的使用场景
- 系统需要复用现有的类,而这些类的接口不符合系统的需求,可以通过适配器模式将其转化为符合需求的类。
- 创建一个通用接口,实现应用程序与第三方库的解耦。无论使用哪种第三方库,应用程序都不需要进行改变
③ 适配器模式的优缺点
- 更好的复用性: 可以将现有的不符合要求的接口进行转化,提高类的复用。
- 用户透明: 客户端可以调用同一接口,而无需因为接口不兼容而改变自身代码。
- 解耦性: 可以用于第三方库与应用程序的解耦,即使更换第三方库,也无需改变应用程序
- 过多的使用适配器,会让系统非常凌乱,不易整体把握。
- 比如,明明看到调用的是A接口,其实内部调用的是B接口。如果系统中很多这样的情况,无异于是一场灾难,还不如直接对系统进行重构。
④ java中适配器模式的应用
- Advice(通知)可以增强被代理类的功能,有不同的类型:
MethodBeforeAdvice
、AfterReturningAdvice
、ThrowsAdvice
。 - 每种类型的Advice都有对应的拦截器,Spring通过适配器模式将Advice封装成对应的拦截器类型,返回给容器。
JDBC
驱动软件是介于JDBC
接口和不同数据库引擎之间的适配器。