1. 程式人生 > 實用技巧 >從頭看看Tomcat啟動Spring容器的原理

從頭看看Tomcat啟動Spring容器的原理

通過帶註解Spring Boot可以啟動一個web容器,並初始化bean容器。那麼Tomcat啟動並初始化spring容器的原理是怎樣的?

Tomcat啟動web程式時會建立一對父子容器(圖1):

有幾種方式:

  1. XML配置Spring和Servlet容器
  2. 通過註解初始化
  3. Servlet提供SPI的呼叫方式來啟動

XML配置Spring和Servlet容器

web.xml主要通過一下配置初始化父子容器(父子容器是職責單一的設計)


<web-app>

    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> <!-- 初始化父容器-->
    </listener>

    <context-param>
        <param-name>contextConfigLocation
</param-name> <param-value>/WEB-INF/app-context.xml</param-value> </context-param> <servlet> <servlet-name>app</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <!-- 初始化子容器-->
<init-param> <param-name>contextConfigLocation</param-name> <param-value></param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet>

Listener主要作用就是初始化spring的bean容器,servlet做request的轉發



<listener> ==> @WebListerner
<Servlet> ==> @WebServlet
<Filter> ==> @WebFilter


Servlet提供SPI的呼叫方式來啟動

開啟service檔案有一行如下:

org.springframework.web.SpringServletContainerInitializer

就是給Tomcat提供初始化容器用的服務介面,Tomcat啟動的時候就會通過ServiceLoader把這個類載入進來,然後呼叫該類的onStartUp方法來初始化
@HandlesTypes({WebApplicationInitializer.class})
public class SpringServletContainerInitializer implements ServletContainerInitializer {

註解中的WebApplicationInitializer.clas就是引入的容器初始化類了。

看一下他的子類:

public abstract class SpringBootServletInitializer implements WebApplicationInitializer {
     public void onStartup(ServletContext servletContext) throws ServletException {
        this.logger = LogFactory.getLog(this.getClass());
        WebApplicationContext rootApplicationContext = this.createRootApplicationContext(servletContext);
        if (rootApplicationContext != null) {
            servletContext.addListener(new SpringBootServletInitializer.SpringBootContextLoaderListener(rootApplicationContext, servletContext));
        } else {
            this.logger.debug("No ContextLoaderListener registered, as createRootApplicationContext() did not return an application context");
        }

    }
}

其中的onStartup方法中有addListener,這個新增的Listener正是對應的XML中:<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>

其中的SpringBootContextLoaderListener是該類的內部類,繼承自ContextLoaderListener

其中的this.createRootApplicationContext(servletContext);便是建立圖1中的父容器