适配器在生活中经常见到,如手机、笔记本电脑的电源适配器,USB 转接头都是常见的适配器。
在设计模式当中,适配器模式既可以作为类结构型模式,也可以作为对象结构型模式。
在类适配器模式中,适配器与适配者之间是继承(实现)关系;在对象适配器模式中,适配器与适配者之间是关联关系。
具体实现
类适配器模式
典型的类适配器模式代码示例如下:
class Adapter extends Adaptee implements Target { public void request() { // 转发调用 super.specificRequest(); } }
从上述的代码可以看出,典型的类适配器模式就是通过继承的方式,使用子类方法调用父类的方法,使用者在使用时面对的是适配器类而不是原始类。
对象适配器模式
典型的对象适配器模式代码示例如下:
class Adapter extends Target { // 维持一个对适配者的引用 private Adaptee adaptee; public Adapter(Adaptee adaptee) { this.adaptee = adaptee; } public void request() { // 转发调用 adaptee.specificRequest(); } }
代理、桥接、装饰器、适配器的区别
笼统地说,这 4 种设计模式可以称为 Wrapper 模式,也就是通过 Wrapper 类二次封装原始类。
尽管它们的代码结构类似,但这 4 种设计模式的用意完全不同,以下是它们简要描述:
- 代理模式:代理类在不改变原始类接口的条件下,为原始类定义一个代理类,主要目的是控制访问,而非加强功能,这是它与装饰器模式最大的不同
- 桥接模式:桥接模式的目的是将接口部分和实现部分分离,从而让它们可以较为容易、也相对独立地加以改变
- 装饰器模式:装饰器模式在不改变原始类接口的情况下,对原始类功能进行加强,并且支持多个装饰器的嵌套使用
- 适配器模式:适配器提供跟原始类不同的接口,而代理模式、装饰器模式提供的都是跟原始类相同的接口,适配器模式是一种事后的补救策略
总结
优点
适配器模式的主要优点如下:
- 将目标类和适配者类解耦
- 增加了类的透明性和复用性
- 灵活性和扩展性都非常好
除上述以外,对象适配器模式还有以下优点:
- 一个对象适配器可以把多个不同的适配器适配到同一个目标
- 可以适配一个适配者的子类
缺点
类适配器模式的主要缺点如下:
- 对于不支持多重继承的编程语言,一次最多只能适配一个适配者类,不能同时适配多个适配者
- 适配者类必须是可被继承的
- 对于不支持多重继承的编程语言,类适配器模式中的目标抽象类只能为接口,不能为类,其使用有一定的局限性
对象适配器模式的主要缺点如下:
- 与类适配器模式相比,要在适配器中置换适配者类的某些方法比较麻烦
适用场景
适配器模式的适用场景如下:
- 封装有缺陷的接口设计
- 统一多个类的接口设计
- 替换依赖的接口设计
- 兼容老版本接口
- 适配不同格式的数据
源码
在 Java 中,SLF4J 日志框架不仅仅提供了统一的接口定义,还提供了针对不同日志框架的适配器。对不同日志框架的接口进行二次封装,适配成统一的 SLF4J 接口定义。