SpringMVC是怎么添加DispatcherServlet到Servlet容器中

1. Servlet與SpringMVC之間的關係

Spring的MVC是基於Servlet功能實現的,經過實現Servlet接口的DispatcherServlet來封裝其核心功能實現。

2. ServletContainerInitializer接口

在web容器啓動時會作一些初始化的工做,例如註冊servlet或者filtes等,servlet規範中經過ServletContainerInitializer實現此功能。

每一個框架, 比如Spring,要使用ServletContainerInitializer就必須在對應的jar包的META-INF/services 目錄建立一個名爲javax.servlet.ServletContainerInitializer的文件,文件內容指定具體的ServletContainerInitializer實現類。(JAVA的SPI特性

案例演示:服務器

@HandlesTypes(value = MyHandlesType.class)//該註解聲明的類,會被注入到set中,若是沒有合適的類,set爲null,这个动作也是Servlet 容器做的(如tomcat)
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");
    }
}

以前传统的Servlet开发,我也需要提供web.xml到Tomcat中,在web.xml中指定Servlet,以及servlet mapping,如:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee 
http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
        version="3.1">

    <!-- spring mvc配置开始  -->
    <servlet>
        <servlet-name>let'sGo</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>let'sGo</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
    <!-- spring mvc配置结束  -->

    <welcome-file-list>
        <welcome-file>index</welcome-file>
    </welcome-file-list>
</web-app>

springmvc是如何實現不需要web.xml配置,靠的就是ServletContainerInitialize。

3. SpringServletContainerInitializer的作用

看下SpringServletContainerInitializer的源码:

@HandlesTypes(WebApplicationInitializer.class)  // Servlet容器会根据@HandlesTypes找到所有注解中的类的实现,也就是所有WebApplicationInitializer.class的实现
public class SpringServletContainerInitializer implements ServletContainerInitializer {
    
    //所有WebApplicationInitializer.class的实现都会被放在这个Set中
   @Override
   public void onStartup(@Nullable Set<Class<?>> webAppInitializerClasses, ServletContext servletContext)
         throws ServletException {

      List<WebApplicationInitializer> initializers = Collections.emptyList();

      if (webAppInitializerClasses != null) {
         initializers = new ArrayList<>(webAppInitializerClasses.size());
         for (Class<?> waiClass : webAppInitializerClasses) {
            // Be defensive: Some servlet containers provide us with invalid classes,
            // no matter what @HandlesTypes says...
            if (!waiClass.isInterface() && !Modifier.isAbstract(waiClass.getModifiers()) &&
                  WebApplicationInitializer.class.isAssignableFrom(waiClass)) {
               try {
                  initializers.add((WebApplicationInitializer)
                        ReflectionUtils.accessibleConstructor(waiClass).newInstance());
               }
               catch (Throwable ex) {
                  throw new ServletException("Failed to instantiate WebApplicationInitializer class", ex);
               }
            }
         }
      }

      if (initializers.isEmpty()) {
         servletContext.log("No Spring WebApplicationInitializer types detected on classpath");
         return;
      }

      servletContext.log(initializers.size() + " Spring WebApplicationInitializers detected on classpath");
      AnnotationAwareOrderComparator.sort(initializers);
      // 调用所有WebApplicationInitializer实现类的onStartUp()函数
      for (WebApplicationInitializer initializer : initializers) {
         initializer.onStartup(servletContext);
      }
   }
}
  • @HandlesTypes(WebApplicationInitializer.class) // Servlet容器会根据@HandlesTypes找到所有注解中的类的实现,也就是所有WebApplicationInitializer.class的实现
  • 所有WebApplicationInitializer.class的实现都会被放在这个Set中
  • 调用所有WebApplicationInitializer实现类的onStartUp()函数,这样Spring框架就可以在WebApplicationInitializer实现类里做一些初始化等操作(如创建DispatcherServlet,添加Filter)

SpringServletContainerInitializer通过实现ServletContainerInitializer将自身并入到Servlet容器的生命周期中, 并通过自身定义的WebApplicationInitializer将依赖于Spring框架的系统初始化需求与Servlet容器解耦. 即依赖于spring的系统可以通过实现WebApplicationInitializer来实现自定义的初始化逻辑. 而不需要去实现ServletContainerInitializer

4. WebApplicationInitializer

SpringServletContainerInitializer会调用所有实现了WebApplicationInitializer接口的实现类。Spring提供了很多实现类用来做初始化操作。

5. AbstractAnnotationConfigDispatcherServletInitializer

SpringMVC提供的AbstractAnnotationConfigDispatcherServletInitializer这个抽象类间接实现了WebApplicationInitializer接口,如下图所示。所以我们只需要自己写一个类,继承AbstractAnnotationConfigDispatcherServletInitializer这个抽象类,servlet容器启动的时候就会调用我们写的实现类的父类里的onStartUp()函数。

image.png

具体其实就是AbstractDispatcherServletInitializer.class的onStartUp()函数:

//org.springframework.web.servlet.support.AbstractDispatcherServletInitializer
public void onStartup(ServletContext servletContext) throws ServletException {
    super.onStartup(servletContext);    
    registerDispatcherServlet(servletContext);
}

protected void registerDispatcherServlet(ServletContext servletContext) {
    String servletName = getServletName();    
    Assert.hasLength(servletName, "getServletName() must not return null or empty");   
    // 创建Spring容器,ServletApplicationContext 
    WebApplicationContext servletAppContext = createServletApplicationContext();    
    Assert.notNull(servletAppContext, "createServletApplicationContext() must not return null");  
    // 创建DispatcherServlet,并把Spring容器当做参数穿进去  
    FrameworkServlet dispatcherServlet = createDispatcherServlet(servletAppContext);    
    Assert.notNull(dispatcherServlet, "createDispatcherServlet(WebApplicationContext) must not return null");    
    dispatcherServlet.setContextInitializers(getServletApplicationContextInitializers());  
    // 将创建好的DispatcherServlet传进Servlet容器里  
    ServletRegistration.Dynamic registration = servletContext.addServlet(servletName, dispatcherServlet);    
    if (registration == null) {
       throw new IllegalStateException("Failed to register servlet with name '" + servletName + "'. " +
             "Check if there is another servlet registered under the same name.");    }

    registration.setLoadOnStartup(1);    
    registration.addMapping(getServletMappings());    
    registration.setAsyncSupported(isAsyncSupported());    
    Filter[] filters = getServletFilters();    
    if (!ObjectUtils.isEmpty(filters)) {
       for (Filter filter : filters) {
          registerServletFilter(servletContext, filter);       }
    }

    customizeRegistration(registration);}

这样的话我们就通过代码的方式做了如下操作:

  1. 创建了Spring容器---ServletApplicationContext
  2. 创建了Servlet---DispatcherServlet,并把Spring容器放进了Servlet中
  3. 把Servlet添加到Servlet 容器中。

6. 上述操作,通过web.xml也能做到,不过Servlet规范3.0以后,支持了代码的方式, 我们看一下通过web.xml执行上述操作的例子:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee 
http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
        version="3.1">

    <!-- spring mvc配置开始  -->
    <servlet>
        <servlet-name>let'sGo</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>let'sGo</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
    <!-- spring mvc配置结束  -->

    <welcome-file-list>
        <welcome-file>index</welcome-file>
    </welcome-file-list>
</web-app>
  1. Servlet容器会根据Servlet规范读取web.xml文件
  2. 读到<servlet>标签后就创建<servlet>里标记的Servlet——这里是DispatcherServlet,这里创建调用的是无参构造函数,跟前面通过代码实现的时候创建DispatcherServlet用的是带参数的构造函数不一样,因为这里没有事先创建好的Spring容器(Servlet WebApplicationContext)
  3. 所以在new DispatcherServlet的过程中,判断当前没有现成的Spring容器(Servlet WebApplicationContext),就会自己创建一个Spring容器,代码如下
//org.springframework.web.servlet.FrameworkServlet,它是DispatcherServlet的父类
protected WebApplicationContext initWebApplicationContext() {
    WebApplicationContext rootContext =
          WebApplicationContextUtils.getWebApplicationContext(getServletContext());    
          WebApplicationContext wac = null;    
          
          if (this.webApplicationContext != null) {
       // A context instance was injected at construction time -> use it
       // 看下构造函数中是否传进来了  webApplicationContext,用web.xml的情况下,webApplicationContext是null     
       wac = this.webApplicationContext;       
       if (wac instanceof ConfigurableWebApplicationContext) {
          ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) wac;          
          if (!cwac.isActive()) {
             // The context has not yet been refreshed -> provide services such as             
             // setting the parent context, setting the application context id, etc             
             if (cwac.getParent() == null) {
                // The context instance was injected without an explicit parent -> set                
                // the root application context (if any; may be null) as the parent                
                cwac.setParent(rootContext);             
                }
             configureAndRefreshWebApplicationContext(cwac);          
             }
       }
    }
    if (wac == null) {
       // No context instance was injected at construction time -> see if one      
        // has been registered in the servlet context. If one exists, it is assumed       
        // that the parent context (if any) has already been set and that the       
        // user has performed any initialization such as setting the context id  
        // 看下Servlet容器里有没有现成的     webApplicationContext
        wac = findWebApplicationContext();    
        }
    // 前面几步都没拿到,自己创建一个webApplicationContext,使用web.xml的情况走的就是这里
    if (wac == null) {
       // No context instance is defined for this servlet -> create a local one      
        wac = createWebApplicationContext(rootContext);    
        }
    ......
}
  • 看下构造函数中是否传进来了 webApplicationContext,用web.xml的情况下,webApplicationContext是null
  • 看下Servlet容器里有没有现成的 webApplicationContext
  • 前面几步都没拿到,自己创建一个webApplicationContext,使用web.xml的情况走的就是这里
  1. 然后还是DispatcherServlet,它会把自己添加到Servlet容器中。
// org.springframework.web.servlet.FrameworkServlet
if (this.publishContext) {
    // Publish the context as a servlet context attribute.    
    String attrName = getServletContextAttributeName();    
    getServletContext().setAttribute(attrName, wac);
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容