• AbstractDispatcherServletInitializer 的实现类为什么可以在初始化Web容器的时候被调用


    引入

    在学习 Spring MVC 引入 Java Web项目的时候,发现除了使用 web.xml 注册 DispatcherServlet 的方式,还可以使用配置类继承 AbstractDispatcherServletInitializer 的方式进行 DispatcherServlet 这个前端控制器的注册。于是好奇我们写的 DispatcherServletInitializer 配置类怎么能被 Web 容器识别并加载。

    解析

    DispatcherServletInitializer 配置类能够被加载的核心是 Java 的 SPI 服务发现机制。

    SPI 服务发现机制

    1. 服务接口 的定义: 某一个框架定义了某种服务的 接口 ,我们在使用这个框架的时候可以使用不同的服务提供商提供的接口的实现。
    2. 注册 服务接口 的实现: 在 classpath:META-INF/services/目录下创建一个以 服务接口 全类名命名的文件,内容为实现类的全类名。
    3. 服务加载 定义了 服务接口 的框架在启动时会到 SPI 规范指定的 classpath:META-INF/services/目录下寻找并加载此框架所需服务。

    Spring MVC 遵循 SPI 规范,作为Servlet容器的服务提供商向其注册服务

    对应于SPI 服务发现机制的三点

    1. 服务接口 的定义 Servlet 规范定义了服务接口 javax-servlet-api-xxx.jar/javax.servlet.ServletContainerInitializer
    2. 注册 服务接口 的实现: Spring MVC 注册服务。在 spring-web-xxx.jar 包下 有 META-INF/services/javax.servlet.ServletContainerInitializer 文件,内容为 org.springframework.web.SpringServletContainerInitializer
    3. 服务加载 SpringServletContainerInitializer 中的 onStartup 方法会在 Servlet 容器初始化的时候通过 SPI 机制发现并调用,onStartup 方法里面又会 将所有 WebApplicationInitializer 的实现类的对象的 onStartup 方法全部调用一遍。文章开头所说的 AbstractDispatcherServletInitializer 也是 WebApplicationInitializer 接口的实现,所以其 onStartup 方法也会在 Servlet 容器初始化的时候被调用。

    展望

    关于 SpringServletContainerInitializer 如何加载 WebApplicationInitializer 还有待研究。

    参考

    Java编程技术之浅析SPI服务发现机制

    附录

    继承 AbstractDispatcherServletInitializer 方式引入 Spring MVC

    public class DispatcherServletInitializer extends AbstractDispatcherServletInitializer {
        @Override
        protected WebApplicationContext createServletApplicationContext() {
            AnnotationConfigWebApplicationContext applicationContext = new AnnotationConfigWebApplicationContext();
            applicationContext.register(SpringMvcConfig.class);
            return applicationContext;
        }
    
        @Override
        protected String[] getServletMappings() {
            // "/" 所有,但不包括.jsp/.html; "/*" 表示包括.jsp/.html 的所有
            return new String[]{"/"};
        }
    
        @Override
        protected WebApplicationContext createRootApplicationContext() {
            return null;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
  • 相关阅读:
    【题解】Educational Codeforces Round 142(CF1792)
    Kubectl 使用详解——k8s陈述式资源管理
    64位操作系统注册ocx控件失败,提示:模块加载失败请确保该二进制存储在指定路径中。
    【部分】CCAA审核员-2022年7月OHSMS职业健康安全体系考试真题
    Java多线程之死锁
    【MongoDB】索引 – 文本索引(指定语言)
    TscanCode的安装与基本使用
    操作系统reboot之后ddr内容还在吗?
    spring cloud 应用框架
    基于石墨烯的光电探测传感器研究
  • 原文地址:https://blog.csdn.net/Nishkata/article/details/125432352