目录
一、简介
适配器模式是一种结构型设计模式,允许将一个类的接口转换成客户端所期望的另一个接口,使得原本由于接口不兼容而不能一起工作的类能够协同工作。适配器模式通常用于连接两个不兼容的接口,允许它们协同工作,而无需修改其原始代码。适配器模式包含以下主要角色:
- 目标接口(Target) : 客户端期待的接口。客户端通过这个接口与适配器进行交互。
- 适配器(Adapter) : 实现了目标接口,并持有一个对被适配者的引用。它将被适配者的接口转换为目标接口,使得被适配者能够被客户端使用。
- 被适配者(Adaptee) : 需要被适配的已存在的类或接口。它是客户端所期望的,但与目标接口不兼容的类。
适配器模式的核心思想是创建一个适配器类,使得客户端可以通过目标接口与适配器进行交互,适配器内部持有一个被适配者对象,并将客户端的请求转发给被适配者,完成请求的转换和适配,使得原本不兼容的接口能够协同工作。
适配器模式的主要用途是在不修改现有代码的情况下,将不兼容的接口进行转换,从而实现系统的兼容性和复用性。通常在集成新旧系统、使用第三方库、或者对现有系统进行扩展时会使用适配器模式。
二、使用
2.1、目标接口
这里我就用我们最常见的场景充电口来举例,我们常见的就是 USB 、 TYPEC ,并且他们都统一了充电的接口,比如都叫 charge 。
java
// 通用的充电接口
public interface Charger {
void charge();
}
// 实现了 USB 接口的 Charger 类
public class USBCharger implements Charger {
public void charge() {
System.out.println("USB接口充电");
}
}
// 实现了 Type-C 接口的 Charger 类
public class TypeCCharger implements Charger {
public void charge() {
System.out.println("Type-C接口充电");
}
}
2.2、被适配者
但是某些厂家就很特殊,它也有充电接口,但是它不叫 charge ,它搞个性叫 chargeWithLightning ,是不是很屌,并且你也不能改它的源码,因为闭源。
java
// 苹果设备充电器接口
public interface LightningCharger {
void chargeWithLightning();
}
// 实现了 LightningCharger 接口的 LightningChargerImpl 类
public class LightningChargerImpl implements LightningCharger {
public void chargeWithLightning() {
System.out.println("Lightning接口充电");
}
}
那怎么搞,现在要统一接口,比如对外都调用 charge ,偏偏某些没源码不能改,不能自己实现,这个时候适配器就很适应了。
2.3、适配器
这个时候我们就给苹果充电增加一个适配器,让适配器实现充电接口,然后适配器调用苹果真实充电,从而对外都是 charge 了。
java
// Lightning 适配到 Charger 的适配器类
public class LightningAdapter implements Charger {
private LightningCharger lightningCharger;
public LightningAdapter(LightningCharger lightningCharger) {
this.lightningCharger = lightningCharger;
}
public void charge() {
lightningCharger.chargeWithLightning();
}
}
2.4、使用
java
public class ChargerAdapterTest {
public static void main(String[] args) {
// 初始化 USB 充电器
Charger usbCharger = new USBCharger();
// 使用 USB 充电器充电
usbCharger.charge();
// 初始化 Type-C 充电器
Charger typeCCharger = new TypeCCharger();
// 使用 Type-C 充电器充电
typeCCharger.charge();
// 创建 Lightning 充电器适配器
LightningCharger lightningCharger = new LightningChargerImpl();
Charger lightningAdapter = new LightningAdapter(lightningCharger);
// 使用 Lightning 适配器充电
lightningAdapter.charge();
}
}
很多人都说,我直接像下面这样实现不就完事了:
java
// 实现了 Type-C 接口的 Charger 类
public class LightningCharger implements Charger {
public void charge() {
System.out.println("Lightning接口充电");
}
}
确实很多场景下,是这样确实是没问的,也应该是这样比较合理。但是我们这里的前提就是假设苹果充电能用,接口方法不一致,但是拿不到苹果代码去改啊。使用适配器的模式总是有些原因的,比如:
- 已存在的代码不允许修改 : 如果原有的接口已经被广泛使用,并且不允许修改其接口定义或者修改成本很高,这时引入适配器模式是一个解决方案。适配器模式可以允许新的类(例如 LightningCharger)与已有接口进行适配,而不需要修改现有代码。
- 接口的变化和未知性 : 在设计初期可能无法预测未来可能的变化。如果在设计阶段就将所有接口都设计得很灵活以适应未来变化,可能会增加不必要的复杂性。适配器模式可以在接口发生变化时,通过适配器来进行兼容,而无需修改大量的现有代码。
- 与第三方库或遗留代码的集成 : 有时我们需要与外部的第三方库或遗留代码进行集成,而这些库或代码的接口可能与我们的系统接口不兼容。为了与它们集成并且使它们能够协同工作,我们可能需要使用适配器模式来适配它们的接口。
- 多个变化源 : 当有多个变化源,例如不同的厂商、不同版本的硬件等,直接修改接口来适应每个变化源可能会导致代码的不稳定性和复杂性增加。适配器模式可以将这些变化源与核心代码解耦,降低了耦合度,使得系统更易于扩展和维护。
总之,适配器模式主要用于将不兼容的接口转换为兼容的接口,以便不同接口的类可以协同工作。在某些情况下,尤其是在面对现有的、不可修改的接口或不可预测的接口变化时,适配器模式可以是一个很有用的解决方案。