国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 學(xué)院 > 開(kāi)發(fā)設(shè)計(jì) > 正文

SpringMVC學(xué)習(xí)筆記(一) DispatcherServlet初始化詳解(應(yīng)用上下文的初始化)

2019-11-11 05:51:34
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

SPRing DispatcherServlet初始化詳解(應(yīng)用上下文的初始化)

Spring的初始化主線(xiàn)如下: 初始化主線(xiàn)圖

DispatcherServlet的繼承體系結(jié)構(gòu)圖如下: DispatcherServlet的繼承體系結(jié)構(gòu)圖

這個(gè)繼承體系結(jié)構(gòu)中HttpServletBean和FrameworkServlet是 對(duì)spring的支持。 HttpServletBean 是 Spring 對(duì)于 Servlet 最低層次的抽象。在這一層抽象中,Spring 會(huì)將這個(gè)Servlet 視作是一個(gè) Spring 的 bean,并將 init-param 中的值作為 bean 的屬性注入進(jìn)來(lái)。 FrameworkServlet 是一個(gè)在其內(nèi)部初始化了一個(gè) Spring 的容器(WebapplicationContext )并暴露了相關(guān)的操作接口, 因而繼承自 FrameworkServlet 的 DispatcherServlet , 也就直接擁有了與 WebApplicationContext 進(jìn)行 通信的能力。DispatcherServlet的繼承體系架起了 DispatcherServlet與 Spring 容器進(jìn)行溝通的橋梁。

從圖中可以看出HttpServletBean繼承HttpServlet,因此在Web容器啟動(dòng)時(shí)將調(diào)用它的init方法,該初始化方法的主要作用: 1. 將Servlet初始化參數(shù)(init-param)設(shè)置到該組件上(如contextAttribute、contextClass、namespace、contextConfigLocation),通過(guò)BeanWrapper簡(jiǎn)化設(shè)值過(guò)程,方便后續(xù)使用; 2. 提供給子類(lèi)初始化擴(kuò)展點(diǎn),initServletBean(),該方法由FrameworkServlet覆蓋。

public abstract class HttpServletBean extends HttpServlet implements EnvironmentAware{ @Override public final void init() throws ServletException { //省略部分代碼 //1、如下代碼的作用是將Servlet初始化參數(shù)設(shè)置到該組件上 //如contextAttribute、contextClass、namespace、contextConfigLocation; try { PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties); BeanWrapper bw = PropertyaccessorFactory.forBeanPropertyAccess(this); ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext()); bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, this.environment)); initBeanWrapper(bw); bw.setPropertyValues(pvs, true); } catch (BeansException ex) { } //2、提供給子類(lèi)初始化的擴(kuò)展點(diǎn),該方法由FrameworkServlet覆蓋 initServletBean(); if (logger.isDebugEnabled()) { logger.debug("Servlet '" + getServletName() + "' configured successfully"); } } //…………省略其他代碼 }

FrameworkServlet繼承HttpServletBean,通過(guò)initServletBean()進(jìn)行Web上下文初始化,該方法主要覆蓋以下兩件事情: 初始化web上下文; 提供給子類(lèi)初始化擴(kuò)展點(diǎn);

public abstract class FrameworkServlet extends HttpServletBean { @Override protected final void initServletBean() throws ServletException { //省略部分代碼 try { //1、初始化Web上下文 this.webApplicationContext = initWebApplicationContext(); //2、提供給子類(lèi)初始化的擴(kuò)展點(diǎn) initFrameworkServlet(); } //省略部分代碼 }}protected WebApplicationContext initWebApplicationContext() { //ROOT上下文(ContextLoaderListener加載的) WebApplicationContext rootContext = WebApplicationContextUtils.getWebApplicationContext(getServletContext()); WebApplicationContext wac = null; if (this.webApplicationContext != null) { // 1、在創(chuàng)建該Servlet注入的上下文 wac = this.webApplicationContext; if (wac instanceof ConfigurableWebApplicationContext) { ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) wac; if (!cwac.isActive()) { if (cwac.getParent() == null) { cwac.setParent(rootContext); } configureAndRefreshWebApplicationContext(cwac); } } } if (wac == null) { //2、查找已經(jīng)綁定的上下文 wac = findWebApplicationContext(); } if (wac == null) { //3、如果沒(méi)有找到相應(yīng)的上下文,并指定父親為ContextLoaderListener wac = createWebApplicationContext(rootContext); } if (!this.refreshEventReceived) { //4、刷新上下文(執(zhí)行一些初始化) onRefresh(wac); } if (this.publishContext) { // Publish the context as a servlet context attribute. String attrName = getServletContextAttributeName(); getServletContext().setAttribute(attrName, wac); //省略部分代碼 } return wac; }

從initWebApplicationContext()方法可以看出,基本上如果ContextLoaderListener加載了上下文將作為根上下文(DispatcherServlet的父容器)。最后調(diào)用了onRefresh()方法執(zhí)行容器的一些初始化,這個(gè)方法由子類(lèi)實(shí)現(xiàn),來(lái)進(jìn)行擴(kuò)展。

DispatcherServlet繼承FrameworkServlet,并實(shí)現(xiàn)了onRefresh()方法提供一些前端控制器相關(guān)的配置

protected void onRefresh(ApplicationContext context) { this.initStrategies(context);}protected void initStrategies(ApplicationContext context) { this.initMultipartResolver(context); this.initLocaleResolver(context); this.initThemeResolver(context); this.initHandlerMappings(context); this.initHandlerAdapters(context); this.initHandlerExceptionResolvers(context); this.initRequestToViewNameTranslator(context); this.initViewResolvers(context); this.initFlashMapManager(context);}

從如上代碼我們可以看出,整個(gè)DispatcherServlet初始化的過(guò)程,具體主要做了如下兩件事情: 1、初始化Spring Web MVC使用的Web上下文,并且可能指定父容器為(ContextLoaderListener加載了根上下 文); 2、初始化DispatcherServlet使用的策略,如HandlerMapping、HandlerAdapter等。

DispatcherServlet的默認(rèn)配置在DispatcherServlet.properties文件中,DispatcherServlet.properties的位置在與DispatcherServlet同一個(gè)包中當(dāng)Spring配置文件中沒(méi)有指定配置時(shí)使用的默認(rèn)策略。 dispatcherServlet配置文件位置圖 具體默認(rèn)配置內(nèi)容如下: dispatcherServlet配置文件默認(rèn)配置 從如上配置可以看出DispatcherServlet在啟動(dòng)時(shí)會(huì)自動(dòng)注冊(cè)這些特殊的Bean,無(wú)需我們注冊(cè),如果我們?cè)O(shè)置注冊(cè)了,默認(rèn)的將不會(huì)注冊(cè)。

DispatcherServlet默認(rèn)使用WebApplicationContext作為上下文,因此我們來(lái)看一下該上下文中特殊的Bean: 1、Controller:處理器/頁(yè)面控制器,做的是MVC中的C的事情,但控制邏輯轉(zhuǎn)移到前端控制器了,用于對(duì)請(qǐng)求進(jìn)行處理; 2、HandlerMapping:請(qǐng)求到處理器的映射,如果映射成功返回一個(gè)HandlerExecutionChain對(duì)象(包含一個(gè)Handler處理器(頁(yè)面控制器)對(duì)象、多個(gè)HandlerInterceptor攔截器)對(duì)象;如BeanNameUrlHandlerMapping將URL與Bean名字映射,映射成功的Bean就是此處的處理器; 3、HandlerAdapter:HandlerAdapter將會(huì)把處理器包裝為適配器,從而支持多種類(lèi)型的處理器,即適配器設(shè)計(jì)模式的應(yīng)用,從而很容易支持很多類(lèi)型的處理器;如SimpleControllerHandlerAdapter將對(duì)實(shí)現(xiàn)了Controller接口的Bean進(jìn)行適配,并且掉處理器的handleRequest方法進(jìn)行功能處理; 4、ViewResolver:ViewResolver將把邏輯視圖名解析為具體的View,通過(guò)這種策略模式,很容易更換其他視圖技術(shù);如InternalResourceViewResolver將邏輯視圖名映射為jsp視圖; 5、LocalResover:本地化解析,因?yàn)镾pring支持國(guó)際化,因此LocalResover解析客戶(hù)端的Locale信息從而方便進(jìn)行國(guó)際化; 6、ThemeResovler:主題解析,通過(guò)它來(lái)實(shí)現(xiàn)一個(gè)頁(yè)面多套風(fēng)格,即常見(jiàn)的類(lèi)似于軟件皮膚效果; 7、MultipartResolver:文件上傳解析,用于支持文件上傳; 8、HandlerExceptionResolver:處理器異常解析,可以將異常映射到相應(yīng)的統(tǒng)一錯(cuò)誤界面,從而顯示用戶(hù)友好的界面(而不是給用戶(hù)看到具體的錯(cuò)誤信息); 9、RequestToViewNameTranslator:當(dāng)處理器沒(méi)有返回邏輯視圖名等相關(guān)信息時(shí),自動(dòng)將請(qǐng)求URL映射為邏輯視圖名; 10、FlashMapManager:用于管理FlashMap的策略接口,F(xiàn)lashMap用于存儲(chǔ)一個(gè)請(qǐng)求的輸出,當(dāng)進(jìn)入另一個(gè)請(qǐng)求時(shí)作為該請(qǐng)求的輸入,通常用于重定向場(chǎng)景。


發(fā)表評(píng)論 共有條評(píng)論
用戶(hù)名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 满洲里市| 白朗县| 吉林市| 中方县| 汤阴县| 云阳县| 元江| 江口县| 武鸣县| 汾西县| 丰都县| 房产| 夏津县| 昆明市| 白朗县| 佛冈县| 阿拉尔市| 天津市| 盖州市| 香河县| 繁峙县| 东平县| 紫金县| 台湾省| 江川县| 鄂尔多斯市| 铁力市| 博乐市| 平顶山市| 泽州县| 白城市| 昌乐县| 濮阳县| 策勒县| 藁城市| 黔江区| 赞皇县| 肥东县| 叶城县| 长子县| 石狮市|