一、了解适配器模式
1.什么是适配器模式
适配器模式将一个类的接口转换成另一种接口,让原本接口不兼容的类可以兼容。这种模式的主要作用就是把原本不兼容的接口,通过适配修改做到统一。
这个模式可以通过创建适配器进行接口转换,让不兼容的接口变得兼容。这可以让客户从实现的接口解耦。如果在一段时间之后,我们想要改变接口,适配器可以将改变的部分封装起来,客户就不必为了应对不同的接口而每次跟着修改。
在业务开发中我们会经常的需要做不同接口的兼容,尤其是中台服务,中台需要把各个业务线的各种类型服务做统一包装,再对外提供接口进行使用。而这在我们平常的开发中也是非常常见的。
适配器模式有两种:“对象”适配器和“类”适配器。
2.UML图解
-
对象适配器
这里写图片描述
这个适配器模式充满着良好的 OO 设计原则:使用对象组合,以修改的接口包装备被适配者;这样做还有一个优点:那就是被适配者的任何子类,都可以搭配着适配器使用。
2.类适配器
这里写图片描述
看起来“类”适配器和“对象”适配器很类似,唯一的差别就是 Adapter 继承了 Target 与 Adaptee。所以“类”适配器必须要使用多重继承才能实现,在 Java 中这是不可能的。所以这里我们主要介绍“对象”适配器。
3.适配器应用场景
-
系统需要使用现有的类,而这些类的接口不符合系统的需要。
-
想要建立一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。
-
需要一个统一的输出接口,而输入端的类型不可预知。
-
二、适配器代码实例
问题描述
以假乱真:如果它走起路来像一只鸭子,叫起来像一只鸭子。那么它可能是一只包装了鸭子类适配器的火鸡。
现在你有一个鸭子 (Duck
) 接口,一个火鸡 (Turkey
) 接口。想让你创建一个鸭子对象,但是现在你只有一只火鸡,怎么办呢?于是你想到用这只火鸡来假扮鸭子。
代码实现 (“对象”适配器模式)
目标鸭子 (Duck
) 接口
package com.jas.adapter;
/**
* 鸭子接口中定义了抽象两个方法,分别是,鸭子“呱呱叫”和鸭子会飞
*/
public interface Duck {
void quack();
void fly();
}
火鸡 (Turkey
) 接口
package com.jas.adapter;
/**
* 火鸡接口中定义了两个抽象方法,分别是火鸡“咯咯叫”和火鸡会飞
*/
public interface Turkey {
void gobble();
void fly();
}
野生火鸡 (WildTurkey
) 类
package com.jas.adapter;
/**
* 实现了 Turkey 接口中的两个方法
*/
public class WildTurkey implements Turkey {
@Override
public void gobble() {
System.out.println("Gobble gobble!");
}
@Override
public void fly() {
System.out.println("I'm flying a short distance!");
}
}
适配器 TurkeyAdapter
类
package com.jas.adapter;
/**
* 适配器类,实现了 Duck 接口,目的将火鸡转换成鸭子,以假乱真
*/
public class TurkeyAdapter implements Duck {
Turkey turkey;
/**
* 通过构造方法,获得适配对象实例化的引用
*
* @param turkey 适配者对象
*/
public TurkeyAdapter(Turkey turkey){
this.turkey = turkey;
}
@Override
public void quack() {
turkey.gobble();
}
@Override
public void fly() {
// 由于火鸡的飞行距离较鸭子近,为了伪装成鸭子,所以让它飞三次来对应鸭子的飞行距离
for (int i = 0; i < 3; i++) {
turkey.fly();
}
}
}
测试 DuckTestDrive
类
package com.jas.adapter;
public class DuckTestDrive {
public static void main(String[] args) {
WildTurkey wildTurkey = new WildTurkey();
// 将火鸡包装进一个火鸡适配器中,使它看起来像一只鸭子
TurkeyAdapter turkeyAdapter = new TurkeyAdapter(wildTurkey);
System.out.println("\nThe TurkeyAdapter says ...");
turkeyAdapter.quack();
turkeyAdapter.fly();
}
}
/**
* 输出
* The TurkeyAdapter says ...
* Gobble gobble!
* I'm flying a short distance!
* I'm flying a short distance!
* I'm flying a short distance!
*
*/
三、实际案例
在业务开发中我们会经常的需要做不同接口的兼容,尤其是中台服务,中台需要把各个业务线的各种类型服务做统一包装,再对外提供接口进行使用。而这在我们平常的开发中也是非常常见的。
åºæ¯æ¨¡æï¼æ¥æ¶å¤ç±»åMQæ¶æ¯
一个系统会接收各种各样的MQ消息或者接口,如果一个个的去开发,就会耗费很大的成本,同时对于后期的拓展也有一定的难度。此时就会希望有一个系统可以配置一下就把外部的MQ接入进行,这些MQ就像上面提到的可能是一些注册开户消息、商品下单消息等等。而适配器的思想方式也恰恰可以运用到这里,适配器不只是可以适配接口往往还可以适配一些属性信息。
四、适配器总结
-
从用户的角度看不到被适配者,是解耦的
-
用户直接调用适配器转化出来的目标接口方法,适配器再调用被适配者的相关接口方法
-
适配器实现了用户收到反馈结果,感觉只是和目标接口交互
-
适配器将一个对象包装起来用以改变其接口。
-
当需要使用一个现有的类而其接口并不符合你的需求时,可以使用适配器。
-
实现一个适配器可能会花一番功夫,可能也会毫不费力,原因是有其目标接口的大小与复杂度决定。
-
适配器模式有两种形式:“类”适配器与“对象适配器”,类适配器需要使用到多重继承,所以在 Java 中并不适用。
-
不使用适配器模式这些功能同样可以实现,但是使用了适配器模式就可以让代码:干净整洁易于维护、减少大量重复的判断和使用、让代码更加易于维护和拓展
更多内容请关注公众号《码上同行》: