傳統tomcat啟動服務與springboot啟動內置tomcat服務的區別(推薦)
spring整合springmvc
- spring整合springmvc中web.xml配置如下,tomcat在啟動過程中會加載web.xml中的內容,ContextLoaderListener實現瞭tomcat裡面的ServletContextListener接口,所以在tomcat容器啟動過程通過ContextLoaderListener來進行spring容器的初始化操作,並將classpath:spring/applicationContext-*.xml指定下的spring配置文件加載,該配置文件我隻配置瞭<context:component-scan base-package=“org.com.yp”/>,代表通過掃描org.com.yp包下的類,包含@Component @Controller@Service等註解等類,進行bean註冊。
- bean註冊是通過AbstractXmlApplicationContext.loadBeanDefinitions該類的方法進行bean定義加載的。
spring中加載bean定義是在org.springframework.context.ConfigurableApplicationContext#refresh方法中的ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory()方法加載bean的,該方法之後會調用org.springframework.context.support.AbstractRefreshableApplicationContext#refreshBeanFactory方法創建bean工廠,並加載的bean定義。
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" id="WebApp_ID" version="3.0"> <display-name>Archetype Created Web Application</display-name> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <!-- 加載spring容器 --> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring/applicationContext-*.xml</param-value> </context-param> <servlet> <servlet-name>mvc-dispatcher</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <!-- 配置springMVC需要加載的配置文件--> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring/spring-*.xml</param-value> </init-param> </servlet> <servlet-mapping> <servlet-name>mvc-dispatcher</servlet-name> <!-- 默認匹配所有的請求 --> <url-pattern>/</url-pattern> </servlet-mapping> </web-app>
當tomcat容器啟動後,通過路徑訪問資源時,第一次會調用org.springframework.web.servlet.HttpServletBean#init方法,之後的http請求就不會再方法該方法類;HttpServletBean實現瞭Servlet接口的規范,所以經過瀏覽器的請求經過servlet接口初始化執行init方法時,會再從spring容器中去加載springmvc配置中定義的加載類,spring與springmvc是父子容器的關系,下面是HttpServletBean的init方法
public final void init() throws ServletException { // Set bean properties from init parameters. PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties); if (!pvs.isEmpty()) { try { BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this); ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext()); bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment())); initBeanWrapper(bw); bw.setPropertyValues(pvs, true); } catch (BeansException ex) { if (logger.isErrorEnabled()) { logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex); } throw ex; } } // 最後會調用org.springframework.context.ConfigurableApplicationContext#refresh容器的刷新方法, // 進行springmvc容器初始化 initServletBean(); } }
springboot啟動容器
- springboot啟動的方式則是先在springboot的org.springframework.boot.SpringApplication#run(java.lang.String…)方法中就初始化瞭spring的上下文環境(裡面包含bean工廠),之後通過org.springframework.boot.SpringApplication#refreshContext方法調用Spring容器中的ConfigurableApplicationContext#refresh方法初始化bean.
- 在spring與springmvc整合的環境中,bean定義的加載是在org.springframework.context.support.AbstractApplicationContext#obtainFreshBeanFactory方法,而springboot中是在
org.springframework.context.support.AbstractApplicationContext#invokeBeanFactoryPostProcessors方法,該方法中通過ConfigurationClassPostProcessor類去加載bean定義,該類實現瞭BeanDefinitionRegistryPostProcessor接口,這個接口允許對bean定義進行加工處理。
// spring中的BeanDefinitionRegistryPostProcessor是BeanFactoryPostProcessor的子接口, // BeanFactoryPostProcessor的作用是在bean的定義信息已經加載但還沒有初始化的時候執行方法postProcessBeanFactory()方法, // 而BeanDefinitionRegistryPostProcessor是在BeanFactoryPostProcessor的前面執行,在源碼 // org.springframework.context.support.PostProcessorRegistrationDelegate#invokeBeanFactoryPostProcessors()方法裡面定義瞭執行順序 // BeanFactoryPostProcessor是bean工廠的bean屬性處理容器,說通俗一些就是可以管理我們的bean工廠內所有的beandefinition(未實例化)數據,可以隨心所欲的修改屬性。 public void refresh() throws BeansException, IllegalStateException { synchronized (this.startupShutdownMonitor) { prepareRefresh(); //獲取告訴子類初始化Bean工廠 將bean加載到緩存中 spring springmvc整合是在這初始化bean的 ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory(); prepareBeanFactory(beanFactory); try { postProcessBeanFactory(beanFactory); // springboot容器啟動加載到這時,初始化瞭下面幾個bean name //0 = "org.springframework.context.annotation.internalConfigurationAnnotationProcessor" =》對應ConfigurationClassPostProcessor類 //1 = "org.springframework.context.annotation.internalAutowiredAnnotationProcessor" =》 AutowiredAnnotationBeanPostProcessor //2 = "org.springframework.context.annotation.internalCommonAnnotationProcessor" =》 CommonAnnotationBeanPostProcessor //3 = "org.springframework.context.event.internalEventListenerProcessor" =》 EventListenerMethodProcessor //4 = "org.springframework.context.event.internalEventListenerFactory" =》 DefaultEventListenerFactory // 調用我們的bean工廠的後置處理器.加載bean定義(不是實例化),通過ConfigurationClassPostProcessor去加載啟動類中的掃描路徑 // 然後將路徑下到bean加載進來 invokeBeanFactoryPostProcessors(beanFactory); registerBeanPostProcessors(beanFactory); initMessageSource(); initApplicationEventMulticaster(); // 這個方法同樣也是留個子類實現的springboot也是從這個方法進行啟動tomat的. onRefresh(); registerListeners(); //實例化我們剩餘的單實例bean. finishBeanFactoryInitialization(beanFactory); // 最後容器刷新 發佈刷新事件(Spring cloud也是從這裡啟動的) finishRefresh(); } catch (BeansException ex) { if (logger.isWarnEnabled()) { logger.warn("Exception encountered during context initialization - " + "cancelling refresh attempt: " + ex); } // Destroy already created singletons to avoid dangling resources. destroyBeans(); // Reset 'active' flag. cancelRefresh(ex); // Propagate exception to caller. throw ex; } finally { // Reset common introspection caches in Spring's core, since we // might not ever need metadata for singleton beans anymore... resetCommonCaches(); } } }
到此這篇關於傳統tomcat啟動服務與springboot啟動內置tomcat服務的區別的文章就介紹到這瞭,更多相關tomcat啟動服務與springboot啟動內置tomcat服務區別內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 關於springmvc報錯404的問題
- SpringMVC執行步驟、Model的使用詳解
- 基於SpringMVC入門案例及講解
- SpringMVC註解的入門實例詳解
- 解決tomcat 靜態頁面(html)中文亂碼的解決終極篇