在講Struts2的初始化以前,應該爲你們描述下Web應用中的過濾器Filter,這關係到咱們對核心過濾器FilterDispatcher的正確理解。java
Filter:一個filter是一個對象,爲每一個請求資源(一個servlet或靜態內容) ,或響應一個資源,或二者,用於執行過濾任務。過濾器執行過濾是在doFilter方法中。每一個過濾器方法訪問一個FilterConfig對象從中獲取初始化參數,filterConfig.getServletContext()能夠得到ServletContext對象使用。過濾器的配置在Web應用程序的web.xml中。web
init():初始化過濾器,它的輸入參數javax.servlet.FilterConfig的一個實例,能夠在這裏初始化過濾要使用到的FilterConfig。該方法由Web容器自動調用。apache
doFilter():進行具體的過濾操做,這個方法以javax.servlet.ServletRequest請求信息, javax.servlet.ServletResponse響應信息,javax.servlet.FilterChain過濾鏈。過濾鏈,在Web應用程序中全部的過濾器會構成一個鏈狀,符合過濾條件的程序將會根據定義的順序執行全部鏈中的過濾器。在這個方法中調用FilterChain的 doFilter(javax.servlet.ServletRequest, javax.servlet.SerletResponse)方法就能夠傳遞到鏈中的下一個過濾器。數組
destory():銷燬過濾器,能夠在這裏釋放使用完的資源,例如設置過濾器中FilterConfig爲null。ide
綜上所述,在Web應用啓動時,會默認初始化Filter,調用Filter的init(FilterConfig filterConfig)方法,當請求到來時,會按順序執行web.xml中所配置Filter的doFilter(ServletRequest req, ServletResponse res, FilterChain chain)方法。源碼分析
Struts2的核心過濾器FilterDispatcher實現的就是StrutsStatics, Filter接口,因此它本質就是一個過濾器,以下圖所示:spa
因此Struts2的初始化工做在Web應用啓動時,就能夠經過FilterDispatcher核心過濾器init(FilterConfig filterConfig)方法來完成了。以下圖所示:插件
FilterDispatcher.init(FilterConfig filterConfig)方法中主要工做分爲:debug
a) 建立Dispatcher類對象,將FilterDispatcher配置的初始化參數傳到該對象中;3d
b) 加載並解析配置文件,配置文件分爲屬性配置文件、Bean配置文件兩種。Struts2的配置文件包括系統默認的配置文件: default.properties、struts-default.xml,以及插件配置文件、應用配置文件:struts-plugin.xml、struts.xml、struts.properties、web.xml。那麼這六種配置文件的加載順序,以下:
1. default.properties
2. struts-default.xml
3. struts-plugin.xml
4. struts.xml
5. struts.properties
6. web.xml
加載順序以下圖所示:
c) 加載靜態資源配置參數: packages,該參數用來配置自動搜尋目錄;
小提示:
FilterDispatcher 實現的StrutsStatics接口,沒有定義業務方法,只定義了若干個常量。Struts2對經常使用的接口進行了從新封裝,好比HttpServletRequest、HttpServletResponse、HttpServletContext等。以下圖所示:
1.2 Struts2初始化源碼分析1. Struts2 Web應用啓動時,根據web.xml配置的核心過濾器FilterDispatcher,會初始化FilterDispatcher:
2. 正如咱們知道的,過濾器初始化時,會自動調用init()方法進行初始化工做,因此在FilterDispatcher啓動時,會自動調用init(FilterConfig filterConfig)方法,進行Struts2的初始化,首先在該方法中會建立org.apache.struts2.Dispatcher對象,將FilterDispatcher配置的初始化參數傳到該對象中,而後調用dispatcher.init()方法加載並解析配置文件,最後加載靜態資源配置參數packages。org.apache.struts2.dispatcher.FilterDispatcher.java源碼以下圖所示:
3. 在FilterDispatcher.init()方法中,首先建立Dispatcher類對象,並將FilterDispatcher配置的初始化參數傳到對象中;相關代碼,以下圖所示:
4. 而後經過dispatcher.init()方法,加載並解析Struts2配置文件,配置文件的加載與解析是由Provider類來實現完成的,因此可分爲兩步:加載配置Provider、解析配置Provider,具體處理步驟以下:
a) 建立com.opensymphony.xwork2.config.ConfigurationManager,其中屬性List<ContainerProvider> containerProviders存放全部配置Provider。
b) init_DefaultProperties():初始化一個用來加載default.properties的DefaultPropertiesProvider,並存入至containerProviders。
c) init_TraditionalXmlConfigurations():默認根據struts-default.xml,struts-plugin.xml,struts.xml (可根據init-param:config 修改加載路徑) 分別建立三個 org.apache.struts2.config. StrutsXmlConfigurationProvider,並存入至containerProviders。
d) init_LegacyStrutsProperties():初始化初始化一個用來加載struts.properties的LegacyPropertiesConfigurationProvider,並存入至containerProviders。
e) init_CustomConfigurationProviders():根據init-param:configProviders初始化一個用戶自定義實現的ConfigurationProvider接口的Provider,並存入至containerProviders。
f) init_FilterInitParameters():初始化一個用來加載web.xml中initParams配置的ConfigurationProvider, 並存入至containerProviders。
g) init_AliasStandardObjects() :初始化一個用來爲所配置的Bean與具體類映射的BeanSelectionProvider,並存入至containerProviders。
h) init_PreloadConfiguration():以上幾步存入ConfigurationProvider對象完畢後,按順序循環調用上面幾步存入的ConfigurationProvider的register、loadPackages、addPackage方法(先加載先解析),進行解析配置Provider。
小提示:
1. 加載配置Provider,其實就是加載配置文件;
2. 解析配置Provider,其實就是解析配置文件;
下面列出以上a-h步的相關代碼,以下圖所示:
1. Struts2中dispatcher.init()代碼:
2. Struts2中init_PreloadConfiguration()方法代碼:
3. XWork中configurationManager.getConfiguration()方法代碼:
4. XWork中configuration.reloadContainer()方法代碼:
5. Xwork中XmlConfigurationProvider.loadPackages()方法代碼:
6. Xwork中XmlConfigurationProvider.addPackage ()方法代碼:
4. 最後經過staticResourceLoader.setHostConfig(new FilterHostConfig(filterConfig))加載靜態資源配置參數:packages,值得注意的是,還有另外三個固定的包和該參數進行拼接,分別是org.apache.struts2.static、template、和org.apache.struts2.interceptor.debugging,中間用空格隔開,通過解析將包名變成路徑後存儲到一個名叫pathPrefixes的數組中,這些目錄中的文件會被自動搜尋;相關代碼,以下圖所示:
注:
關於源碼分析,大概分爲兩種:流程源碼分析、過程源碼分析,因本人的初衷是流程源碼分析,因此以上的分析是根據初始化處理流程順序來進行分析的,並未對各個方法的過程細節作深刻的講解,望見諒。