适配器的单词是
Adapter
,我们在开发时经常碰到叫做XxxAdapter
的类,此时一般就是使用了适配器模式,适配器模式是非常常用,本文就对适配器模式做一个简单的介绍
假设现系统有一个日志接口,但是其实现性能比较低下,开发者想换一个高性能的日志框架时,遇到了新的框架和原先的接口不兼容的问题,这个时候应该怎么办?如何让新的日志框架适配原先的日志接口。
其实在不考虑设计模式的情况下,也可以想到解决方式,那就是在原先的接口实现中,使用新框架的接口和方法实现原来的功能。
但是因为使用的是框架,所以修改之前的实现不是个好办法,比较麻烦,此时就可以考虑适配器模式。
适配器模式就是在两个层不兼容是采取的方案,简单说就是再加一个层。
对于图中的Client
,原先使用的FormerImpl
的实例,现在想使用Adaptee
的对象,没法直接使用因为接口已经固定了,代码中也都使用的是FormerImpl
,不可能再去修改代码,所以定义一个Adapter对象,作为一个中间层,向Client提供Adaptee的功能。
Target.java
:public interface LogFace {
void logInfo(String info);
}
FormerImpl.java
:public class FormerImpl implements LogFace{
@Override
public void logInfo(String info) {
System.out.println(info);
}
}
Adaptee.java
:@Slf4j
public class Adaptee {
public void logInfo(String info){
log.info("高性能日志实现"+info);
}
}
Adapter.java
:public class Adapter implements LogFace{
private Adaptee adaptee = new Adaptee();
@Override
public void logInfo(String info) {
adaptee.logInfo(info);
}
}
LogFactory.java
:使用工厂方法获取日志门对象public class LogFactory {
//更换日志实现,只需要在此处返回new Adapter的对象返回即可
private static LogFace logFace = new Adapter();
public static LogFace getInstance(){
return logFace;
}
}
Client.java
:public class Client {
public static void main(String[] args) {
LogFace logFace = LogFactory.getInstance();
logFace.logInfo("打印日志");
}
}
注意:在实例代码中添加一个工厂方法,用于创建日志门面接口(LogFace)的实现对象,与更换日志实现之前相比,客户端代码没有任何改变。
见示例代码。
适配器模式一个最典型的应用就是在SpringMVC中,HandlerAdapter就是扮演的适配器的角色。
当收到客户端请求时,DispatcherServlet首先根据请求的URL等从处理器映射器(HandlerMapping)中找是否有对应的处理器,然后交给对应的HandlerAdapter去执行其中的处理请求的方法。其中:
注意:
这里的模式使用思路和上面说的不太一样,这里的Adaptee有很多(Controller一般是有很多),所以更多是为了适配不同的Controller去进行统一的处理器方法执行,和JDBC接口一样的作用。
但是本质就是加一层,屏蔽差异。所以本身面向接口编程就有点适配器模式的味道在里面了。
主要解决在软件系统中,常常要将一些"现存的对象"放到新的环境中,而新环境要求的接口是现对象不能满足的。
参考:
[1] 一次性搞懂设计模式–适配器模式 - 掘金 (juejin.cn)
[2] 适配器模式 | 菜鸟教程 (runoob.com)
[3] 秒懂设计模式之适配器模式(Adapter Pattern) - 知乎 (zhihu.com)
[4] Spring中的设计模式之适配器模式 - 知乎 (zhihu.com)