定义
适配器模式(Adapter Pattern):将一个类的接口转换成客户端希望的另一个接口。适配器模式让那些接口不兼容的类可以一起工作。
模式结构与实现
- Target(目标抽象类):目标抽象类定义客户端所需要的的接口,可以是一个抽象类或接口,也可以是具体类。在适配器中,由于Java语言不支持多继承,它只能是接口。
- Adapter(适配器类):它可以调用另一个接口,作为一个转换器,对Adaptee和Target进行适配。适配器Adapter是适配器模式的核心,在类适配器中,它通过实现Target接口并继承Adaptee类来使二者产生联系,在对象适配器中,它通过继承Target并关联一个Adaptee对象使二者产生联系。
- Adaptee(适配者类):适配者即被适配的角色,它定义了一个已经存在的接口,这个接口需要适配,适配者类般是一个具体类,包含了客户希望使用的业务方法,在某些情况下甚至没有适配者类的源代码。
代码
类适配器
Adaptee
public class Adaptee {
public void specificRequest() {
System.out.println("适配者类");
}
}
Target
public interface Target {
void request();
}
Adapter
public class Adapter extends Adaptee implements Target {
@Override
public void request() {
super.specificRequest();
}
}
调用
public class Test {
public static void main(String[] args) {
Target target = new Adapter();
target.request();
}
}
由于Java只支持单继承,所以Target(目标抽象类)只能是接口interface
对象适配器
public class Adapter1 implements Target {
private Adaptee adaptee;
public Adapter1(Adaptee adaptee) {
this.adaptee = adaptee;
}
@Override
public void request() {
adaptee.specificRequest();
}
}
在这里,Target(目标抽象类)可以设置为class了。
缺省适配器模式
- ServiceInterface(适配者接口):它是一个接口,通常在该接口中声明了大量的方法。
- AbstractServiceClass(缺省适配器类):它是缺省适配器模式的核心,使用空方法的形式实现在在ServiceInterface接口中声明的方法。通常将它定义为抽象类,因为对它进行实例化没有任何意义。
- ConcreteServiceClass(具体业务类):它是缺省适配器的子类,在没有引入适配器之前它需要实现适配者接口,因此需要实现适配者接口中定义的所有方法,而对于一些无需使用的方法不得不提供空实现。在有类缺省适配器之后可以直接继承缺省适配器,根据需要有选择性地覆盖适配器类中定义的方法。
代码如下
public abstract class AbstractServiceClass implements ServiceInterface {
public void serviceMethod1() {} // 空方法
public void serviceMethod2() {} // 空方法
public void serviceMethod3() {} // 空方法
}
双向适配器
public class Adaptor implements Traget, Adaptee {
// 同时维护对抽象目标类和适配者的引用
private Target target;
private Adaptee adaptee;
public Adaptor(Target target) {
this.target = target;
}
public Adaptor(Adaptee adaptee) {
this.adaptee = adaptee;
}
@Override
public void request() {
adaptee.specificRequest();
}
@Override
public void specificRequest() {
target.request();
}
}
优/缺点与适用环境
- 优点
- 将目标类和适配者类解耦,通过引人一个适配器类来重用现有的适配者类,无须修改原有结构。
- 增加了类的透明性和复用性,将具体的业务实现过程封装在适配者类中,对于客户端类而言是透明的,而且提高了适配者的复用性,同一个适配者类可以在多个不同的系统中复用。
- 灵活性和扩展性都非常好,通过使用配置文件可以很方便地更换适配器,也可以在不修改原有代码的基础上增加新的适配器类,完全符合开闭原则。
具体来说,类适配器模式还有以下优点:
由于适配器类是适配者类的子类,因此可以在适配器类中置换一些适配者的方法,使得适配器的灵活性更强。
对象适配器模式还有以下优点:
- 一个对象适配器可以把多个不同的适配者适配到同一个目标。
- 可以适配一个适配者的子类,由于适配器和适配者之间是关联关系,根据里氏代换原则,适配者的子类也可通过该适配器进行适配。
- 缺点
- 对于Java、C#等不支持多重类继承的语言,一次最多只能适配一个适配者类,不能同时适配多个适配者。
- 适配者类不能为最终类,例如在Java中不能为final类。
- 在Java、C#等语言中.类适配器模式中的目标抽象类只能为接口,不能为类,其使用有一定的局限性。
对象适配器模式的缺点主要如下:
与类适配器模式相比,在该模式下要在适配器中置换适配者类的某些方法比较麻烦。如果一定要置换掉适配者类的一个或多个方法,可以先做一个适配者类的子类,将适配者类的方法置换掉,然后再把适配者类的子类当成真正的适配者进行适配,实现过程较为复杂。
- 适用环境
- 系统需要使用一些现有的类,而这些类的接口(例如方法名)不符合系统的需要,甚至没有这此类的源代码。
- 想创建一个可以重复使用的类,用于和一些彼此之间没有太大关联的类(包括一些可能在将来引进的类)一起工作。