亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

淺談Spring與SpringMVC父子容器的關(guān)系與初始化

 更新時間:2020年08月27日 11:27:25   作者:菜到懷疑人生  
這篇文章主要介紹了淺談Spring與SpringMVC父子容器的關(guān)系與初始化,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

Spring和SpringMVC的容器具有父子關(guān)系,Spring容器為父容器,SpringMVC為子容器,子容器可以引用父容器中的Bean,而父容器不可以引用子容器中的Bean。

了解了Spring與SpringMVC父子容器的關(guān)系,接下來讓我們看看Spring與SpringMVC容器的初始化過程。

以下講解使用的web.xml文件如下:

 <context-param>
    <param-name>contextConfigLocation</param-name>//指定spring ioc配置文件的位置
    <param-value>classpath*:spring/*.xml</param-value>
  </context-param>
  <!-- Creates the Spring Container shared by all Servlets and Filters -->
  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
<!-- 配置DisaptcherServlet -->
  <servlet>
    <servlet-name>springMVC</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <!-- 初始化參數(shù),配置springmvc配置文件 -->
    <init-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>springMVC配置文件的路徑</param-value>
    </init-param>
    <!-- web容器啟動時加載該Servlet -->
    <load-on-startup>1</load-on-startup>
  </servlet>
  <servlet-mapping>
    <servlet-name>springMVC</servlet-name>
    <url-pattern>/</url-pattern>
  </servlet-mapping>

spring ioc容器初始化的過程

1、web應用程序啟動時,tomcat會讀取web.xml文件中的context-parm(含有配置文件的路徑)和listener節(jié)點,接著會為應用程序創(chuàng)建一個ServletContext,為全局共享,Spring ioc容器就是存儲在這里

2、tomcat將context-param節(jié)點轉(zhuǎn)換為鍵值對,寫入到ServletContext中

3、創(chuàng)建listener節(jié)點中的ContextLoaderListener實例,調(diào)用該實例,初始化webapplicationContext,這是一個接口,其實現(xiàn)類為XmlWebApplicationContext(即spring的IOC容器),其通過ServletContext.getinitialParameter("contextConfigLoaction")從ServletContext中獲取context-param中的值(即spring ioc容器配置文件的路徑),這就是為什么要有第二步的原因。接著根據(jù)配置文件的路徑加載配置文件信息(其中含有Bean的配置信息)到WebApplicationContext(即spring ioc容器)中,將WebApplicationContext以WebApplicationContext.ROOTWEBAPPLICATIONCONTEXTATTRIBUTE為屬性Key,將其存儲到ServletContext中,便于獲取。至此,spring ioc容器初始化完畢

4、容器初始化web.xml中配置的servlet,為其初始化自己的上下文信息servletContext,并加載其設(shè)置的配置信息到該上下文中。將WebApplicationContext(即spring ioc容器)設(shè)置為它的父容器。其中便有SpringMVC(假設(shè)配置了SpringMVC),這就是為什么spring ioc是springmvc ioc的父容器的原因

SpringMVC初始化過程

SpringMVC通過web.xml文件中servlet標簽下的DispatcherServlet類完成自身的初始化

DispatcherServlet類的繼承體系如下:

請注意每個長方形中第三行的方法,其為完成SpringMVC ioc容器初始化的關(guān)鍵。

我們知道,每個servlet在初始化時,會先調(diào)用servlte的構(gòu)造函數(shù)(為默認構(gòu)造函數(shù)),接著調(diào)用init函數(shù),而DispatcherServlet的init方法在其父類HttpServlet中。

HttpServlet中的init方法

/DispatcherServlet第一次加載時調(diào)用init方法
@Override
  public final void init() throws ServletException {
    if (logger.isDebugEnabled()) {
      logger.debug("Initializing servlet '" + getServletName() + "'");
    }
    // Set bean properties from init parameters.
    try {
/*加載web.xml文件中的servlet標簽中的init-param,其中含有springMVC的配置文件的名字和路徑
 *若沒有,則默認為(servlet-name)-servlet.xml,
 *默認路徑為WEF—INF下
 */
      PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);
     //創(chuàng)建BeanWrapper實例,為DispatcherServlet設(shè)置屬性
      BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);
      ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext());
      bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment()));
      initBeanWrapper(bw);
     //把init-param中的參數(shù)設(shè)置到DispatcherServlet里面去
      bw.setPropertyValues(pvs, true);
    }
    catch (BeansException ex) {
      logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex);
      throw ex;
    }
 
 
    // Let subclasses do whatever initialization they like.
    //該方法在FrameworkServlet中
    initServletBean();
 
 
    if (logger.isDebugEnabled()) {
      logger.debug("Servlet '" + getServletName() + "' configured successfully");
    }
  }

FrameworkServlet中的initServletBean方法

@Override
  protected final void initServletBean() throws ServletException {
    getServletContext().log("Initializing Spring FrameworkServlet '" + getServletName() + "'");
    if (this.logger.isInfoEnabled()) {
      this.logger.info("FrameworkServlet '" + getServletName() + "': initialization started");
    }
    long startTime = System.currentTimeMillis();
 
    try {
      //創(chuàng)建springmvc的ioc容器實例
      this.webApplicationContext = initWebApplicationContext();
      initFrameworkServlet();
    }
    catch (ServletException ex) {
      this.logger.error("Context initialization failed", ex);
      throw ex;
    }
    catch (RuntimeException ex) {
      this.logger.error("Context initialization failed", ex);
      throw ex;
    }
 
    if (this.logger.isInfoEnabled()) {
      long elapsedTime = System.currentTimeMillis() - startTime;
      this.logger.info("FrameworkServlet '" + getServletName() + "': initialization completed in " +
          elapsedTime + " ms");
    }
  }

FrameworkServlet中的initWebapplicationContext方法

protected WebApplicationContext initWebApplicationContext() {
    //首先通過ServletContext獲得spring容器,因為子容器springMVC要和父容器spring容器進行關(guān)聯(lián)
    //這就是為什么要在ServletContext中注冊spring ioc容器的原因
    WebApplicationContext rootContext =
        WebApplicationContextUtils.getWebApplicationContext(getServletContext());
    //定義springMVC容器wac
    WebApplicationContext wac = null;
 
 
    //判斷容器是否由編程式傳入(即是否已經(jīng)存在了容器實例),存在的話直接賦值給wac,給springMVC容器設(shè)置父容器
    //最后調(diào)用刷新函數(shù)configureAndRefreshWebApplicationContext(wac),作用是把springMVC的配置信息加載到容器中去(之前已經(jīng)將配置信息的路徑設(shè)置到了bw中)
    if (this.webApplicationContext != null) {
      // A context instance was injected at construction time -> use it
      wac = this.webApplicationContext;
      if (wac instanceof ConfigurableWebApplicationContext) {
        ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) wac;
        if (!cwac.isActive()) {
 
 
          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
            //將spring ioc設(shè)置為springMVC ioc的父容器
            cwac.setParent(rootContext);
          }
 
 
          configureAndRefreshWebApplicationContext(cwac);
        }
      }
    }
    if (wac == null) {
      // 在ServletContext中尋找是否有springMVC容器,初次運行是沒有的,springMVC初始化完畢ServletContext就有了springMVC容器
      wac = findWebApplicationContext();
    }
 
 
    //當wac既沒有沒被編程式注冊到容器中的,也沒在ServletContext找得到,此時就要新建一個springMVC容器
    if (wac == null) {
      // 創(chuàng)建springMVC容器
      wac = createWebApplicationContext(rootContext);
    }
 
 
    if (!this.refreshEventReceived) {
      //到這里mvc的容器已經(jīng)創(chuàng)建完畢,接著才是真正調(diào)用DispatcherServlet的初始化方法onRefresh(wac)
      onRefresh(wac);
    }
 
 
    if (this.publishContext) {
      //將springMVC容器存放到ServletContext中去,方便下次取出來
      String attrName = getServletContextAttributeName();
      getServletContext().setAttribute(attrName, wac);
      if (this.logger.isDebugEnabled()) {
        this.logger.debug("Published WebApplicationContext of servlet '" + getServletName() +
            "' as ServletContext attribute with name [" + attrName + "]");
      }
    }
    return wac;
  }

FrameworkServlet中的createWebApplicationContext(WebApplicationContext parent)方法

protected WebApplicationContext createWebApplicationContext(ApplicationContext parent) {
    Class<?> contextClass = getContextClass();
    if (this.logger.isDebugEnabled()) {
      this.logger.debug("Servlet with name '" + getServletName() +
          "' will try to create custom WebApplicationContext context of class '" +
          contextClass.getName() + "'" + ", using parent context [" + parent + "]");
    }
    if (!ConfigurableWebApplicationContext.class.isAssignableFrom(contextClass)) {
      throw new ApplicationContextException(
          "Fatal initialization error in servlet with name '" + getServletName() +
          "': custom WebApplicationContext class [" + contextClass.getName() +
          "] is not of type ConfigurableWebApplicationContext");
    }
    //實例化空白的ioc容器
    ConfigurableWebApplicationContext wac =
        (ConfigurableWebApplicationContext) BeanUtils.instantiateClass(contextClass);
    //給容器設(shè)置環(huán)境
    wac.setEnvironment(getEnvironment());
    //給容器設(shè)置父容器(就是spring容器),兩個ioc容器關(guān)聯(lián)在一起了
    wac.setParent(parent);
    //給容器加載springMVC的配置信息,之前已經(jīng)通過bw將配置文件路徑寫入到了DispatcherServlet中
    wac.setConfigLocation(getContextConfigLocation());
    //上面提到過這方法,刷新容器,根據(jù)springMVC配置文件完成初始化操作,此時springMVC容器創(chuàng)建完成
    configureAndRefreshWebApplicationContext(wac);
 
    return wac;
  }

DispatcherServlet的onRefresh(ApplicationContext context)方法

@Override
  protected void onRefresh(ApplicationContext context) {
    initStrategies(context);
  }

DispatcherServlet的initStrategies(ApplicationContext context)方法

protected void initStrategies(ApplicationContext context) {
    initMultipartResolver(context);//文件上傳解析
    initLocaleResolver(context);//本地解析
    initThemeResolver(context);//主題解析
    initHandlerMappings(context);//url請求映射
    initHandlerAdapters(context);//初始化真正調(diào)用controloler方法的類
    initHandlerExceptionResolvers(context);//異常解析
    initRequestToViewNameTranslator(context);
    initViewResolvers(context);//視圖解析
    initFlashMapManager(context);
  }

總結(jié)以下DispatcherServlet及各個父類(接口)的功能:

HttpServlet:實現(xiàn)了init方法,完成web,xml中與DispatcherServlet有關(guān)的參數(shù)的讀入,初始化DispatcherServlet。

FrameworkServlet:完成了springMVC ioc 容器的創(chuàng)建,并且將spring ioc容器設(shè)置為springMVC ioc容器的父容器,將springMVC ioc容器注冊到ServletContext中

DispatcherServlet:完成策略組件的初始化

至此,SpringMVC容器初始化完成

以上這篇淺談Spring與SpringMVC父子容器的關(guān)系與初始化就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • 解決IDEA Maven下載依賴時報錯ERROR - #org.jetbrains.idea.maven - Cannot reconnect.

    解決IDEA Maven下載依賴時報錯ERROR - #org.jetbrains.ide

    這篇文章主要介紹了解決IDEA Maven下載依賴時報錯ERROR - #org.jetbrains.idea.maven - Cannot reconnect.問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • 解決SpringBoot webSocket 資源無法加載、tomcat啟動報錯的問題

    解決SpringBoot webSocket 資源無法加載、tomcat啟動報錯的問題

    這篇文章主要介紹了解決SpringBoot webSocket 資源無法加載、tomcat啟動報錯的問題,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-11-11
  • Spring security中的授權(quán)

    Spring security中的授權(quán)

    本篇為大家?guī)鞸pring security的授權(quán),首先要理解一些概念,有關(guān)于:權(quán)限、角色、安全上下文、訪問控制表達式、方法級安全性、訪問決策管理器,這篇文章主要介紹了Spring security中的授權(quán),需要的朋友可以參考下
    2024-01-01
  • java偽泛型知識點詳解

    java偽泛型知識點詳解

    在本篇文章里小編給大家整理的是一篇關(guān)于java偽泛型知識點詳解內(nèi)容,有需要的朋友們可以跟著學習參考下。
    2021-06-06
  • 一步步教你把SpringBoot項目打包成Docker鏡像

    一步步教你把SpringBoot項目打包成Docker鏡像

    Docker可以讓開發(fā)者打包他們的應用以及依賴包到一個輕量級、可移植的容器中,然后發(fā)布到任何流行的 Linux 機器上,也可以實現(xiàn)虛擬化,下面這篇文章主要給大家介紹了關(guān)于SpringBoot項目打包成Docker鏡像的相關(guān)資料,需要的朋友可以參考下
    2023-02-02
  • Java多線程中常見的幾個問題

    Java多線程中常見的幾個問題

    這篇文章主要介紹了Java多線程中常見的幾個問題 ,需要的朋友可以參考下
    2015-05-05
  • java實現(xiàn)找出兩個文件中相同的單詞(兩種方法)

    java實現(xiàn)找出兩個文件中相同的單詞(兩種方法)

    這篇文章主要介紹了java實現(xiàn)找出兩個文件中相同的單詞(兩種方法),需要的朋友可以參考下
    2020-08-08
  • 使用Java實現(xiàn)動態(tài)生成MySQL數(shù)據(jù)庫

    使用Java實現(xiàn)動態(tài)生成MySQL數(shù)據(jù)庫

    這篇文章主要為大家詳細介紹了如何使用Java實現(xiàn)動態(tài)生成MySQL數(shù)據(jù)庫,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2024-02-02
  • Spring?Boot配置內(nèi)容加密實現(xiàn)敏感信息保護

    Spring?Boot配置內(nèi)容加密實現(xiàn)敏感信息保護

    之前我們講過的配置相關(guān)知識都是Spring?Boot原生就提供的,而今天我們將介紹的功能并非Spring?Boot原生就支持,但卻非常有用:配置內(nèi)容的加密
    2021-11-11
  • 詳解Alibaba?Java診斷工具Arthas查看Dubbo動態(tài)代理類

    詳解Alibaba?Java診斷工具Arthas查看Dubbo動態(tài)代理類

    這篇文章主要介紹了Alibaba?Java診斷工具Arthas查看Dubbo動態(tài)代理類?,它可以幫助我們查看JDK或者javassist生成的動態(tài)代理類,當然,它的功能遠不止此,還可以在生產(chǎn)環(huán)境進行診斷,需要的朋友可以參考下
    2022-04-04

最新評論