Spring web容器
Spring MVC項目啟動時會有兩個ApplicationContext容器扼脐,Root ApplicationContext用于管理Service層及以下層的bean,而dispacherServlet ApplicationContext用于管理Controller層的bean。
web.xml
<!DOCTYPE web-app PUBLIC
"-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
"http://java.sun.com/dtd/web-app_2_3.dtd" >
<web-app>
<display-name>應用名稱</display-name>
<!-- 用于spring容器初始化時讀取配置 -->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath*:META-INF/spring/applicationContext-*.xml</param-value>
</context-param>
<!-- 創(chuàng)建Spring容器 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<!-- SpringMVC前端控制器 -->
<servlet>
<servlet-name>DispatcherServlet</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 用于spring 控制器容器配置 -->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>WEB-INF/spring/webmvc-config.xml</param-value>
</init-param>
<init-param>
<param-name>contextClass</param-name>
<param-value>
org.springframework.web.context.support.AnnotationConfigWebApplicationContext
</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>DispatcherServlet</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
解析:
web.xml中注冊了ContextLoaderListener實現(xiàn)了 ServletContextListener
ServletContextListener接口有如下方法力惯,會在servlet容器初始化和銷毀的時候回調(diào)
public interface ServletContextListener extends EventListener {
void contextInitialized(ServletContextEvent var1);
void contextDestroyed(ServletContextEvent var1);
}
監(jiān)聽器會在servlet容器啟動和銷毀時回調(diào)以上兩個方法放棒。
ContextLoaderListener就是在contextInitialized方法中創(chuàng)建、初始化Root ApplicationContext的站欺。
1.Root ApplicationContext容器啟動
(1)ContextLoaderListener 方式:
繼承了ContextLoader
servlet容器如tomcat啟動時回調(diào)contextInitialized()
ContextLoaderListener.contextInitialized(event)
ContextLoader.initWebApplicationContext(event.getServletContext());
ContextLoader.createWebApplicationContext(sc)// 從web.xml中獲取contextClass參數(shù)(上文的AnnotationConfigWebApplicationContext)姨夹,找到相應類,實例化,并設(shè)置其ServletContext
由此創(chuàng)建了 Root WebApplicationContext矾策。
(2)WebApplicationInitializer方式:
實現(xiàn)WebApplicationInitializer
public class MyWebApplicationInitializer implements WebApplicationInitializer {
@Override
public void onStartup(ServletContext servletCxt) {
// 創(chuàng)建AC并加載配置
AnnotationConfigWebApplicationContext ac = new AnnotationConfigWebApplicationContext();
ac.register(AppConfig.class);
ac.refresh();//abstractRefrshAC 創(chuàng)建BeanFactory并創(chuàng)建Bean和裝配
// 創(chuàng)建 DispatcherServlet 并注冊
DispatcherServlet servlet = new DispatcherServlet(ac);
ServletRegistration.Dynamic registration = servletCxt.addServlet("app", servlet);
registration.setLoadOnStartup(1);
registration.addMapping("/app/*");
// Servlet容器啟動的時候會初始化DispatcherServlet.init會啟動子AC
}
}
啟動流程:
StandardContext是tomcat的容器組件磷账,是servlet容器,在tomcat啟動時會頂級容器Server會掌管子容器的生命周期贾虽,層層啟動逃糟,Servlet 3.0規(guī)范在StandardContext.startInternal()時會查找啟動Spring相關(guān)的,如下調(diào)用關(guān)系:
StandardContext.startInternal()
// servlet 3.0容器 容器啟動時會查找ServletContainerInitializer這個接口的實現(xiàn)類啟動
SpringServletContainerInitializeron.Startup()
// 實現(xiàn)ServletContainerInitializer蓬豁,會查找WebApplicationInitializer實現(xiàn)類
XXWebApplicationInitializer.Startup()
2.DispatcherServlet實例化+初始化
DispatcherServlet由tomcat時啟動容器管理其生命周期調(diào)用init()方法初始化
繼承FrameworkServlet,FrameworkServlet繼承ServletBean
DispatcherServlet.init() //繼承自祖父HttpServletBean
FrameworkServlet.initServletBean()
FrameworkServlet.initWebApplicationContext()
FrameworkServlet.createWebApplicationContext()//創(chuàng)建AC并設(shè)置parent
FrameworkServlet.configureAndRefreshWebApplicationContext()
wac.refresh();
DispatcherServlet.onRefresh(webac)
DispatcherServlet.initMultipartResolver(context);
DispatcherServlet.initLocaleResolver(context);
DispatcherServlet.initThemeResolver(context);
DispatcherServlet.initHandlerMappings(context);
DispatcherServlet.initHandlerAdapters(context);
DispatcherServlet.initHandlerExceptionResolvers(context);
DispatcherServlet.initRequestToViewNameTranslator(context);
DispatcherServlet.initViewResolvers(context);
DispatcherServlet.initFlashMapManager(context);
如此創(chuàng)建好了
4.為什么要有兩個ApplicationContext
(1)隔離 由于管理Controller層的子AC可以委托雙親 ApplicationContext去查找bean绰咽,所以Root ApplicationContext容器中的Bean是共享的,而子AC中的Bean卻不能被管理下層Service層和DAO層的Root Application獲取地粪,從而達到上層依賴下層的純粹性取募。