Spring MVC攔截器_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
Spring為我們提供了:
org.springframework.web.servlet.HandlerInterceptor接口,
org.springframework.web.servlet.handler.HandlerInterceptorAdapter適配器,
實(shí)現(xiàn)這個(gè)接口或繼承此類(lèi),可以非常方便的實(shí)現(xiàn)自己的攔截器。
有以下三個(gè)方法:
Action之前執(zhí)行:
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler);
生成視圖之前執(zhí)行
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView);
最后執(zhí)行,可用于釋放資源
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
分別實(shí)現(xiàn)預(yù)處理、后處理(調(diào)用了Service并返回ModelAndView,但未進(jìn)行頁(yè)面渲染)、返回處理(已經(jīng)渲染了頁(yè)面)
在preHandle中,可以進(jìn)行編碼、安全控制等處理;
在postHandle中,有機(jī)會(huì)修改ModelAndView;
在afterCompletion中,可以根據(jù)ex是否為null判斷是否發(fā)生了異常,進(jìn)行日志記錄。
參數(shù)中的Object handler是下一個(gè)攔截器。
如何使用攔截器?
自定義一個(gè)攔截器,要實(shí)現(xiàn)HandlerInterceptor接口:
Java代碼
public class MyInteceptor implements HandlerInterceptor { 略。。。 }
Spring MVC并沒(méi)有總的攔截器,不能對(duì)所有的請(qǐng)求進(jìn)行前后攔截。
Spring MVC的攔截器,是屬于HandlerMapping級(jí)別的,可以有多個(gè)HandlerMapping ,每個(gè)HandlerMapping可以有自己的攔截器。
當(dāng)一個(gè)請(qǐng)求按Order值從小到大,順序執(zhí)行HandlerMapping接口的實(shí)現(xiàn)類(lèi)時(shí),哪一個(gè)先有返回,那就可以結(jié)束了,后面的HandlerMapping就不走了,本道工序就完成了。就轉(zhuǎn)到下一道工序了。
攔截器會(huì)在什么時(shí)候執(zhí)行呢? 一個(gè)請(qǐng)求交給一個(gè)HandlerMapping時(shí),這個(gè)HandlerMapping先找有沒(méi)有處理器來(lái)處理這個(gè)請(qǐng)求,如何找到了,就執(zhí)行攔截器,執(zhí)行完攔截后,交給目標(biāo)處理器。
如果沒(méi)有找到處理器,那么這個(gè)攔截器就不會(huì)被執(zhí)行。
在spring MVC的配置文件中配置有三種方法:
方案一,(近似)總攔截器,攔截所有url
Java代碼
<mvc:interceptors> <bean class="com.app.mvc.MyInteceptor" /> </mvc:interceptors>
為什么叫“近似”,前面說(shuō)了,Spring沒(méi)有總的攔截器。
<mvc:interceptors/>
會(huì)為每一個(gè)HandlerMapping,注入一個(gè)攔截器??傆幸粋€(gè)HandlerMapping是可以找到處理器的,最多也只找到一個(gè)處理器,所以這個(gè)攔截器總會(huì)被執(zhí)行的。起到了總攔截器的作用。
如果是REST風(fēng)格的URL,靜態(tài)資源也會(huì)被攔截。
方案二, (近似) 總攔截器, 攔截匹配的URL。
Xml代碼
<mvc:interceptors > <mvc:interceptor> <mvc:mapping path="/user/*" /> <!-- /user/* --> <bean class="com.mvc.MyInteceptor"></bean> </mvc:interceptor> </mvc:interceptors>
就是比 方案一多了一個(gè)URL匹配。
如果是REST風(fēng)格的URL,靜態(tài)資源也會(huì)被攔截。
方案三,HandlerMappint上的攔截器。
如果是REST風(fēng)格的URL,靜態(tài)資源就不會(huì)被攔截。因?yàn)槲覀兙珳?zhǔn)的注入了攔截器。
Xml代碼
<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"> <property name="interceptors"> <list> <bean class="com.mvc.MyInteceptor"></bean> </list> </property> </bean>
如果使用了<mvc:annotation-driven />,
它會(huì)自動(dòng)注冊(cè)DefaultAnnotationHandlerMapping 與AnnotationMethodHandlerAdapter 這兩個(gè)bean,所以就沒(méi)有機(jī)會(huì)再給它注入interceptors屬性,就無(wú)法指定攔截器。
當(dāng)然我們可以通過(guò)人工配置上面的兩個(gè)Bean,不使用 <mvc:annotation-driven />,就可以 給interceptors屬性 注入攔截器了。
其實(shí)我也不建議使用 <mvc:annotation-driven />,
而建議手動(dòng)寫(xiě)詳細(xì)的配置文件,來(lái)替代 <mvc:annotation-driven />
,這就控制力就強(qiáng)了。
如何替換 <mvc:annotation-driven />
?他到底做了什么工作?
一句 <mvc:annotation-driven />
實(shí)際做了以下工作:(不包括添加自己定義的攔截器)
我們了解這些之后,對(duì)Spring3 MVC的控制力就更強(qiáng)大了,想改哪就改哪里。
Xml代碼
<!-- 注解請(qǐng)求映射 --> <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"> <property name="interceptors"> <list> <ref bean="logNDCInteceptor"/> <!-- 日志攔截器,這是你自定義的攔截器 --> <ref bean="myRequestHelperInteceptor"/> <!-- RequestHelper攔截器,這是你自定義的攔截器--> <ref bean="myPermissionsInteceptor"/> <!-- 權(quán)限攔截器,這是你自定義的攔截器--> <ref bean="myUserInfoInteceptor"/> <!-- 用戶信息攔截器,這是你自定義的攔截器--> </list> </property> </bean> <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"> <property name="messageConverters"> <list> <ref bean="byteArray_hmc" /> <ref bean="string_hmc" /> <ref bean="resource_hmc" /> <ref bean="source_hmc" /> <ref bean="xmlAwareForm_hmc" /> <ref bean="jaxb2RootElement_hmc" /> <ref bean="jackson_hmc" /> </list> </property> </bean> <bean id="byteArray_hmc" class="org.springframework.http.converter.ByteArrayHttpMessageConverter" /><!-- 處理.. --> <bean id="string_hmc" class="org.springframework.http.converter.StringHttpMessageConverter" /><!-- 處理.. --> <bean id="resource_hmc" class="org.springframework.http.converter.ResourceHttpMessageConverter" /><!-- 處理.. --> <bean id="source_hmc" class="org.springframework.http.converter.xml.SourceHttpMessageConverter" /><!-- 處理.. --> <bean id="xmlAwareForm_hmc" class="org.springframework.http.converter.xml.XmlAwareFormHttpMessageConverter" /><!-- 處理.. --> <bean id="jaxb2RootElement_hmc" class="org.springframework.http.converter.xml.Jaxb2RootElementHttpMessageConverter" /><!-- 處理.. --> <bean id="jackson_hmc" class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter" /><!-- 處理json-->
相關(guān)文章
Spring boot validation校驗(yàn)方法實(shí)例
這篇文章主要給大家介紹了關(guān)于Spring boot validation校驗(yàn)方法的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02Kotlin語(yǔ)言編程Regex正則表達(dá)式實(shí)例詳解
這篇文章主要為大家介紹了Kotlin語(yǔ)言編程Regex正則表達(dá)式實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08Spring MVC請(qǐng)求參數(shù)的獲取教程指南
本文介紹了SpringMVC中如何獲取各種類(lèi)型的請(qǐng)求參數(shù),包括基本類(lèi)型、POJO、數(shù)組、集合以及RESTful風(fēng)格的參數(shù),還討論了請(qǐng)求參數(shù)中文亂碼的解決方案,參數(shù)綁定的注解如@RequestParam,以及自定義類(lèi)型轉(zhuǎn)換器的實(shí)現(xiàn),需要的朋友可以參考下2024-10-10JSP頁(yè)面?zhèn)鲄⒊霈F(xiàn)中文亂碼的解決方案
這篇文章主要介紹了JSP頁(yè)面?zhèn)鲄⒊霈F(xiàn)中文亂碼的解決方案,非常實(shí)用,需要的朋友可以參考下2014-08-08SpringValidation自定義注解及分組校驗(yàn)功能詳解
這篇文章主要介紹了SpringValidation自定義注解及分組校驗(yàn)功能,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2024-01-01IntelliJ IDEA中Scala、sbt、maven配置教程
這篇文章主要介紹了IntelliJ IDEA中Scala、sbt、maven配置教程,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09Spring多定時(shí)任務(wù)@Scheduled執(zhí)行阻塞問(wèn)題解決
這篇文章主要介紹了Spring多定時(shí)任務(wù)@Scheduled執(zhí)行阻塞問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05