SpringMVC源码深度解析之SpringServletContainerInitializer原理分析

SpringMVC

对SpringMVC或者其它比较成熟的MVC框架而言,解决的问题无外乎如下几点:html

  • 将web页面的请求传给服务器
  • 根据不一样的请求处理不一样的逻辑单元
  • 返回处理结果数据并跳转至响应的页面

Servlet与SpringMVC之间的关系

Spring的MVC是基于Servlet功能实现的,经过实现Servlet接口的DispatcherServlet来封装其核心功能实现。java

快速搭建Servlet环境

<dependencies>
  <!-- 添加Servlet支持 -->
  <dependency>
    <groupId>javax.servlet</groupId>
    <artifactId>javax.servlet-api</artifactId>
    <version>3.1.0</version>
  </dependency>
  <dependency>
    <groupId>javax.servlet.jsp</groupId>
    <artifactId>javax.servlet.jsp-api</artifactId>
    <version>2.3.1</version>
  </dependency>
</dependencies>

servlet线程是否安全?

不安全,构造函数只执行一次web

ServletContainerInitializer接口

在web容器启动时会作一些初始化的工做,例如注册servlet或者filtes等,servlet规范中经过ServletContainerInitializer实现此功能。spring

每一个框架要使用ServletContainerInitializer就必须在对应的jar包的META-INF/services 目录建立一个名为javax.servlet.ServletContainerInitializer的文件,文件内容指定具体的ServletContainerInitializer实现类。api

一、Servlet容器启动会扫描,当前应用里面每个jar包的ServletContainerInitializer的实现
二、提供ServletContainerInitializer的实现类;必须绑定在META-INF/services/javax.servlet.ServletContainerInitializer,文件的内容就是ServletContainerInitializer实现类的全类名;
tomcat

META-INF:tomcat默认就会去读取的安全

案例演示:服务器

@HandlesTypes(value = MyHandlesType.class)//该注解声明的类,会被注入到set中,若是没有合适的类,set为null
public class MyServletContainerInitializer implements ServletContainerInitializer {

    /**
     * @param set 感兴趣类型 也就是MyHandlesType 全部子类型
     * @param servletContext
     * @throws ServletException
     */
    public void onStartup(Set<Class<?>> set, ServletContext servletContext) throws ServletException {
        // 1.打印全部感兴趣的类型
        for (Class<?> c : set) {
            System.out.println(c);
        }
       // 2.servletContext 手动注册过滤器、servlet、监听器
        ServletRegistration.Dynamic payServlet = servletContext.addServlet("payServlet", new PayServlet());
        payServlet.addMapping("/pay");
    }
}

基于注解方式构建SpringMVC框架

<dependencies>
  <dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-webmvc</artifactId>
    <version>5.0.5.RELEASE</version>
  </dependency>
  <!-- 添加Servlet支持 -->
  <dependency>
    <groupId>javax.servlet</groupId>
    <artifactId>javax.servlet-api</artifactId>
    <version>3.1.0</version>
  </dependency>
  <dependency>
    <groupId>javax.servlet.jsp</groupId>
    <artifactId>javax.servlet.jsp-api</artifactId>
    <version>2.3.1</version>
  </dependency>
</dependencies>

springmvc是如何实现无web.xml配置,靠的就是ServletContainerInitializermvc

SpringServletContainerInitializer的做用:加载一些第三方的依赖信息oracle

@RestController
public class IndexController {

    @RequestMapping(value = "/",produces="text/html;charset=UTF-8")
    public String index() {
        return "success...";
    }
    /**
     * springmvc环境的时候须要配置那些东西?
     * SpringMVC启动的时候如何实现没有web.xml
     */
}
@Configuration
@ComponentScan("com.mayikt.controller")
@EnableWebMvc
public class SpringMvcConfig {
    //@EnableWebMvc 等于开启SpringMVC注解方式
    //@Configuration xml
    // @ComponentScan("com.mayikt.controller") mvc 扫包范围
    //DispatcherServlet
}
public class WebInitializer implements WebApplicationInitializer {
    @Override
    public void onStartup(ServletContext servletContext) throws ServletException {
       // 1.启动SpringMVC 容器 类注入到Spring中
        AnnotationConfigWebApplicationContext app = new AnnotationConfigWebApplicationContext();// 启动SpringMVC Web
        // 2.注入咱们的springmvc 的配置文件
        app.register(SpringMvcConfig.class);
        // 3. 将咱们的DispatcherServlet 注入到 serlvet容器中
        ServletRegistration.Dynamic dynamic = servletContext.addServlet("dispatcher", new DispatcherServlet(app));
        // 4.填写url路径映射
        dynamic.addMapping("/");
        dynamic.setLoadOnStartup(1);// 优先级最高表示 最先被加载
    }
  // 基本配置已经ok呢? web.xml ? 使用WebApplicationInitializer 替代web.xml //为何这个类WebInitializer 不须要注解呢? 可以自动的找到该类呢?
}

SpringServletContainerInitializer源码分析

SpringServletContainerInitializer:

Servlet3.0引入的接口,用在web应用启动时动态添加servlet、filter和listener

基于spi机制,META-INF/services/javax.servlet.ServletContainerInitializer文件中存放实现该接口的类,这些类会被容器调用;

只能使用在jar文件中,不能在web项目中使用。

进入:org.springframework.web.SpringServletContainerInitializer
@HandlesTypes({WebApplicationInitializer.class})
public class SpringServletContainerInitializer implements ServletContainerInitializer {
    public SpringServletContainerInitializer() {
    }
    public void onStartup(@Nullable Set<Class<?>> webAppInitializerClasses, ServletContext servletContext) throws ServletException {
...
}

SpringServletContainerInitializer 为spring中实现ServletContainerInitializer接口的惟一类,该类主要是从容器获取实现WebApplicationInitializer的类,而且按照次序javax.annotation.Priority)调用其onStartup方法

咱们定义的WebInitializer实现了WebApplicationInitializer接口

public class WebInitializer implements WebApplicationInitializer {
    public void onStartup(javax.servlet.ServletContext servletContext) throws ServletException {
  ....
    }
}

原理:实现了WebApplicationInitializer 接口的类会被Spring初始化;因为web项目中不能使用spi机制(未定位),因此若是须要动态添加servlet、filter和listener,就能够实现该接口,交由spring初始化。

springmvc是如何替代web.xml

使用:SpringServletContainerInitializer 提供给SpringMVC实现初始化

debug:能够看到WebApplicationInitializer子类有咱们自定义的WebInitializer

本文参考:

蚂蚁课堂:http://www.mayikt.com/

文档https://docs.oracle.com/javaee/7/api/javax/servlet/ServletContainerInitializer.html?is-external=true

相关文章
相关标签/搜索