一、介绍
1)适配器模式(Adapter Pattern)将某个类的接口转换成客户端期望的另一个接口表示,主的目的是兼容性,让原本因接口不匹配不能一起工作的两个类可以协同工作。其别名为包装器(Wrapper)
2)适配器模式属于结构型模式
3)主要分为三类:类适配器模式、对象适配器模式、接口适配器模式
二、类适配器模式
基本介绍
Adapter类通过继承src类,实现dst类接口,完成src -> dst的适配
注意事项
1)Java是单继承机制,所以类适配器需要继承src类这一点算是一个缺点,因为这要求dst必须是接口,有一定局限性
2)src类的方法在Adapter中都会暴露出来,也增加了使用的成本
3)由于其继承了src类,所以它可以根据需求重写src类的方法,使得Adapter的灵活性增强了
三、对象适配器模式
1)基本思路和类的适配器模式相同,只是将Adapter类作修改,不是继承src类,而是持有src类的实例,以解决兼容性的问题。即:持有src类,实现dst类接口,完成src->dst的适配
2)根据“合成复用原则”,在系统中尽量使用关联关系来替代继承关系。
3)对象适配器模式是适配器模式常用的一种
四、接口适配器模式
1)一些书籍称为:适配器模式(Default Adapter Pattern)或缺省适配器模式。
2)当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆盖父类的某些方法来实现需求
3)适用于一个接口不想使用其所有的方法的情况。
五、SpringMVC框架应用的源码分析
SpringMVC中的HandlerAdapter就使用了适配器模式
Controller
/**
* @Date Created in 2022/9/20 15:13
* @Description 多种Controller实现
*/
public interface Controller {
}
class HttpController implements Controller{
public void doHttpHandler(){
System.out.println("doHttpHandler......");
}
}
class SimpleController implements Controller{
public void doSimplerHandler(){
System.out.println("doSimplerHandler......");
}
}
class AnnotationController implements Controller{
public void doAnnotationHandler(){
System.out.println("doAnnotationHandler......");
}
}
HandlerAdapter适配器
/**
* @Date Created in 2022/9/20 15:15
* @Description 适配器
*/
public interface HandlerAdapter {
boolean support(Object handler);
void handle(Object handler);
}
class HttpHandlerAdapter implements HandlerAdapter{
@Override
public boolean support(Object handler) {
return (handler instanceof HttpController);
}
@Override
public void handle(Object handler) {
((HttpController) handler).doHttpHandler();
}
}
class SimpleHandlerAdapter implements HandlerAdapter{
@Override
public boolean support(Object handler) {
return (handler instanceof SimpleController);
}
@Override
public void handle(Object handler) {
((SimpleController) handler).doSimplerHandler();
}
}
class AnnotationHandlerAdapter implements HandlerAdapter{
@Override
public boolean support(Object handler) {
return (handler instanceof AnnotationController);
}
@Override
public void handle(Object handler) {
((AnnotationController) handler).doAnnotationHandler();
}
}
模拟DispatchServlet
/**
* @Date Created in 2022/9/20 15:19
* @Description
*/
public class DispatchServlet {
public static List<HandlerAdapter> handlerAdapters = new ArrayList<>();
public DispatchServlet(){
handlerAdapters.add(new HttpHandlerAdapter());
handlerAdapters.add(new SimpleHandlerAdapter());
handlerAdapters.add(new AnnotationHandlerAdapter());
}
public void doDispatch(){
// 此处模拟springmvc从request取handler对象
// 因为每个controller的处理方法名称不同,
// 采用适配器能够统一调用不同controller的目标方法
Controller controller = new HttpController();
HandlerAdapter adapter = getHandler(controller);
adapter.handle(controller);
}
public HandlerAdapter getHandler(Controller controller) {
for (HandlerAdapter adapter : handlerAdapters) {
if(adapter.support(controller)){
return adapter;
}
}
return null;
}
public static void main(String[] args) {
new DispatchServlet().doDispatch();
}
}
执行结果
doHttpHandler…
六、注意事项
1)三种命名方式,是根据 src是以怎样的形式给到Adapter(在Adapter里的形式)来命名的。
2)类适配器:以类给到,在Adapter里,就是将src当做类,继承
对象适配器:以对象给到,在Adapter里,将src作为一个对象,持有
接口适配器:以接口给到,在Adapter里,将src作为一个接口,实现
3)Adapter模式最大的作用还是将原本不兼容的接口融合在一起工作。
4)实际开发中,实现起来不拘泥于我们讲解的三种经典形式