啟動(dòng)Spring項(xiàng)目詳細(xì)過(guò)程(小結(jié))
1、Spring 項(xiàng)目放到web項(xiàng)目容器中(Tomcat、Jetty、JBoss)
本文以通用的Tomcat為例
2、項(xiàng)目容器啟動(dòng)時(shí)需要加載讀取web.xml配置文件
如下圖:
3、容器首先會(huì)去讀取web.xml配置文件中的兩個(gè)節(jié)點(diǎn):<listener> </listener>和<context-param> </context-param>
說(shuō)明:
tomcat在啟動(dòng)web容器的時(shí)候會(huì)啟動(dòng)一個(gè)叫ServletContextListener的監(jiān)聽器,每當(dāng)在web容器中有ServletContextListener這個(gè)接口被實(shí)例化的時(shí)候,web容器會(huì)通知ServletContextListener被實(shí)例的對(duì)象去執(zhí)行其contextInitialized()的方法進(jìn)行相應(yīng)的業(yè)務(wù)處理;
而spring框架在設(shè)計(jì)的過(guò)程中ContextLoadListener這個(gè)類實(shí)現(xiàn)了ServletContextListener這個(gè)接口,因此每當(dāng)有ContextLoadListener這個(gè)類被實(shí)例化的時(shí)候,web容器會(huì)通知Spring執(zhí)行contextInitialized()這個(gè)方法,從而進(jìn)行spring容器的啟動(dòng)與創(chuàng)建的過(guò)程中;
4、ContextLoaderListener中的contextInitialized()進(jìn)行了spring容器的啟動(dòng)配置,調(diào)用initWebApplicationContext初始化spring容器;
@Override public void contextInitialized(ServletContextEvent event) { initWebApplicationContext(event.getServletContext()); }
public WebApplicationContext initWebApplicationContext(ServletContext servletContext) { //Spring 啟動(dòng)的句柄,spring容器開始啟動(dòng)的根目錄 if(servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE) != null) { throw new IllegalStateException("Cannot initialize context because there is already a root application context present - check whether you have multiple ContextLoader* definitions in your web.xml!"); } else { Log logger = LogFactory.getLog(ContextLoader.class); servletContext.log("Initializing Spring root WebApplicationContext"); if(logger.isInfoEnabled()) { logger.info("Root WebApplicationContext: initialization started"); } long startTime = System.currentTimeMillis(); try { //處理spring容器是否已經(jīng)創(chuàng)建(只創(chuàng)建沒(méi)有創(chuàng)建spring的各個(gè)bean) if(this.context == null) { this.context = this.createWebApplicationContext(servletContext); } if(this.context instanceof ConfigurableWebApplicationContext) { ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext)this.context; if(!cwac.isActive()) { if(cwac.getParent() == null) { ApplicationContext parent = this.loadParentContext(servletContext); cwac.setParent(parent); } //Spring容器創(chuàng)建完成后,加載spring容器的各個(gè)組件 this.configureAndRefreshWebApplicationContext(cwac, servletContext); } } servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context); ClassLoader ccl = Thread.currentThread().getContextClassLoader(); if(ccl == ContextLoader.class.getClassLoader()) { currentContext = this.context; } else if(ccl != null) { currentContextPerThread.put(ccl, this.context); } if(logger.isDebugEnabled()) { logger.debug("Published root WebApplicationContext as ServletContext attribute with name [" + WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE + "]"); } if(logger.isInfoEnabled()) { long elapsedTime = System.currentTimeMillis() - startTime; logger.info("Root WebApplicationContext: initialization completed in " + elapsedTime + " ms"); } return this.context; } catch (RuntimeException var8) { logger.error("Context initialization failed", var8); servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, var8); throw var8; } catch (Error var9) { logger.error("Context initialization failed", var9); servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, var9); throw var9; } } }
5、spring容器創(chuàng)建完成后,準(zhǔn)備開始實(shí)例化加載bean,Spring容器創(chuàng)建完成后,準(zhǔn)備向spring容器中加載bean 使用configureAndRefreshWebApplicationContext(cwac, servletContext); 完成bean的加載;
protected void configureAndRefreshWebApplicationContext(ConfigurableWebApplicationContext wac, ServletContext sc) { if (ObjectUtils.identityToString(wac).equals(wac.getId())) { // The application context id is still set to its original default value // -> assign a more useful id based on available information String idParam = sc.getInitParameter(CONTEXT_ID_PARAM); if (idParam != null) { wac.setId(idParam); } else { // Generate default id... wac.setId(ConfigurableWebApplicationContext.APPLICATION_CONTEXT_ID_PREFIX + ObjectUtils.getDisplayString(sc.getContextPath())); } } wac.setServletContext(sc); String configLocationParam = sc.getInitParameter(CONFIG_LOCATION_PARAM); if (configLocationParam != null) { wac.setConfigLocation(configLocationParam); } // The wac environment's #initPropertySources will be called in any case when the context // is refreshed; do it eagerly here to ensure servlet property sources are in place for // use in any post-processing or initialization that occurs below prior to #refresh ConfigurableEnvironment env = wac.getEnvironment(); if (env instanceof ConfigurableWebEnvironment) { ((ConfigurableWebEnvironment) env).initPropertySources(sc, null); } customizeContext(sc, wac); wac.refresh(); }
說(shuō)明:
configureAndRefreshWebApplicationContext中加載spring的配置文件,即web.xml中讀取<context-param></context-param>中加載到Spring的配置文件,即:classpath:/config/applicationContext.xml;
或
通過(guò)以下代碼加載spring配置
public class Application{ public static void main(String[] args) { ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("/context.xml"); ctx.start(); } }
此處略過(guò)如何調(diào)用DefaultResourceLoader
頂級(jí)接口ResourceLoader僅提供了一個(gè)getResource(String location)方法,可以根據(jù)一個(gè)資源地址加載資源文件,資源地址的表達(dá)式可以是以下幾種:
--1. classpath:前綴開頭的表達(dá)式,例如: classpath:smart-context.xml
--2.“/”開頭的表達(dá)式,例如:/WEB-INF/classes/smart-context.xml
--3. 非“/”開頭的表達(dá),例如:WEB-INF/classes/smart-context.xml
--4. url協(xié)議,例如:file:/D:/ALANWANG-AIA/Horse-workspace/chapter3/target/classes/smart-context.xml
Spring提供了實(shí)現(xiàn)類DefaultResourceLoader,DefaultResourceLoader在實(shí)現(xiàn)了以上列舉的功能基礎(chǔ)上,還為開發(fā)者提供了自定義擴(kuò)展接口ProtocolResolver,開發(fā)者可實(shí)現(xiàn)該接口定制個(gè)性化資源表達(dá)式,代碼如下:
@Override public Resource getResource(String location) { Assert.notNull(location, "Location must not be null"); for (ProtocolResolver protocolResolver : this.protocolResolvers) { // 1 Resource resource = protocolResolver.resolve(location, this); if (resource != null) {return resource;} } if (location.startsWith("/")) {return getResourceByPath(location);} //2 else if (location.startsWith(CLASSPATH_URL_PREFIX)) { //3 return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length()), getClassLoader()); } else { try { // Try to parse the location as a URL... URL url = new URL(location); //4 return new UrlResource(url); } catch (MalformedURLException ex) { // No URL -> resolve as resource path. return getResourceByPath(location); //5 } } }
步驟1,先用擴(kuò)展協(xié)議解析器解析資源地址并返回。舉個(gè)例子,咱們可以自定義資源解析器來(lái)完成帶前綴“classpath:”的解析:
首先實(shí)現(xiàn)ProtocolResolver接口:
class ClasspathPreProtocolResolver implements ProtocolResolver{ private static String CLASS_PATH_PRE="classpath:"; public Resource resolve(String location, ResourceLoader resourceLoader) { if( location.startsWith(CLASS_PATH_PRE)) { return new ClassPathResource(location.substring(CLASS_PATH_PRE.length())); } return null; } }
步驟2,假設(shè)location以斜杠開頭,則調(diào)用該類中 getResourceByPath(String path)方法 ,代碼如下:
protected Resource getResourceByPath(String path) { return new ClassPathContextResource(path, getClassLoader()); }
步驟三,假如資源表達(dá)式以classpath開頭,則截取除前綴calsspath:的路徑,并做為ClassPathResource的構(gòu)造參數(shù),生成ClassPathResource實(shí)例后返回。咱們可以在web.xml中做如下配置:
<context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:/config/applicationContext.xml</param-value> </context-param>
6、通過(guò)refresh()
內(nèi)部的實(shí)現(xiàn)我們大致可以了解整個(gè)refresh()
方法擔(dān)負(fù)了整個(gè)Spring容器初始化和加載的所有邏輯,包括Bean工廠的初始化、post-processor的注冊(cè)以及調(diào)用、bean的實(shí)例化、事件發(fā)布等。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot解決jar包沖突的問(wèn)題,簡(jiǎn)單有效
這篇文章主要介紹了SpringBoot解決jar包沖突的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09Java微信授權(quán)登陸的實(shí)現(xiàn)示例
微信授權(quán)登錄,官方文檔寫的比較簡(jiǎn)潔。所以對(duì)于會(huì)的人一目了然,而對(duì)于新手剛?cè)腴T的人來(lái)說(shuō)是舉步維艱。本文詳細(xì)的介紹了Java微信授權(quán)登陸的實(shí)現(xiàn)示例,感興趣的朋友可以了解一下2021-06-06Java實(shí)現(xiàn)飛機(jī)大戰(zhàn)-II游戲詳解
《飛機(jī)大戰(zhàn)-II》是一款融合了街機(jī)、競(jìng)技等多種元素的經(jīng)典射擊手游。游戲是用java語(yǔ)言實(shí)現(xiàn),采用了swing技術(shù)進(jìn)行了界面化處理,感興趣的可以了解一下2022-02-02Java基礎(chǔ)之?dāng)?shù)組超詳細(xì)知識(shí)總結(jié)
這篇文章主要介紹了Java基礎(chǔ)之?dāng)?shù)組詳解,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有很好的幫助,需要的朋友可以參考下2021-05-05Java基于ShardingSphere實(shí)現(xiàn)分庫(kù)分表的實(shí)例詳解
ShardingSphere?已于2020年4月16日成為?Apache?軟件基金會(huì)的頂級(jí)項(xiàng)目,?它們均提供標(biāo)準(zhǔn)化的數(shù)據(jù)水平擴(kuò)展、分布式事務(wù)和分布式治理等功能,可適用于如?Java?同構(gòu)、異構(gòu)語(yǔ)言、云原生等各種多樣化的應(yīng)用場(chǎng)景,對(duì)ShardingSphere分庫(kù)分表相關(guān)知識(shí)感興趣的朋友一起看看吧2022-03-03JAVA正則表達(dá)式校驗(yàn)qq號(hào)碼的方法
Java作為一種開發(fā)語(yǔ)言,有許多值得推薦的地方,但是它一直以來(lái)沒(méi)有自帶對(duì)正則表達(dá)式的支持。下面小編給大家?guī)?lái)了JAVA正則表達(dá)式校驗(yàn)qq號(hào)碼的方法,需要的朋友參考下吧2018-04-04Java+Spring+MySql環(huán)境中安裝和配置MyBatis的教程
這篇文章主要介紹了Java+Spring+MySql環(huán)境中安裝和配置MyBatis的教程,MyBatis一般被用來(lái)增強(qiáng)數(shù)據(jù)庫(kù)操作,文中對(duì)MyBatis的主配置文件有較為詳細(xì)的講解,需要的朋友可以參考下2016-04-04SpringBoot實(shí)現(xiàn)文件上傳下載功能小結(jié)
最近做的一個(gè)項(xiàng)目涉及到文件上傳與下載功能。SpringBoot后臺(tái)如何實(shí)現(xiàn)文件上傳下載呢?下面有單文件上傳和多文件上傳功能,感興趣的朋友一起看看吧2017-08-08