您好,登錄后才能下訂單哦!
Spring為我們提供了:
org.springframework.web.servlet.HandlerInterceptor接口,
org.springframework.web.servlet.handler.HandlerInterceptorAdapter適配器,
實現這個接口或繼承此類,可以非常方便的實現自己的攔截器。
有以下三個方法:
Action之前執行:
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler);
生成視圖之前執行
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView);
最后執行,可用于釋放資源
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
分別實現預處理、后處理(調用了Service并返回ModelAndView,但未進行頁面渲染)、返回處理(已經渲染了頁面)
在preHandle中,可以進行編碼、安全控制等處理;
在postHandle中,有機會修改ModelAndView;
在afterCompletion中,可以根據ex是否為null判斷是否發生了異常,進行日志記錄。
參數中的Object handler是下一個攔截器。
如何使用攔截器?
自定義一個攔截器,要實現HandlerInterceptor接口:
Java代碼
public class MyInteceptor implements HandlerInterceptor { 略。。。 }
Spring MVC并沒有總的攔截器,不能對所有的請求進行前后攔截。
Spring MVC的攔截器,是屬于HandlerMapping級別的,可以有多個HandlerMapping ,每個HandlerMapping可以有自己的攔截器。
當一個請求按Order值從小到大,順序執行HandlerMapping接口的實現類時,哪一個先有返回,那就可以結束了,后面的HandlerMapping就不走了,本道工序就完成了。就轉到下一道工序了。
攔截器會在什么時候執行呢? 一個請求交給一個HandlerMapping時,這個HandlerMapping先找有沒有處理器來處理這個請求,如何找到了,就執行攔截器,執行完攔截后,交給目標處理器。
如果沒有找到處理器,那么這個攔截器就不會被執行。
在spring MVC的配置文件中配置有三種方法:
方案一,(近似)總攔截器,攔截所有url
Java代碼
<mvc:interceptors> <bean class="com.app.mvc.MyInteceptor" /> </mvc:interceptors>
為什么叫“近似”,前面說了,Spring沒有總的攔截器。
<mvc:interceptors/>
會為每一個HandlerMapping,注入一個攔截器。總有一個HandlerMapping是可以找到處理器的,最多也只找到一個處理器,所以這個攔截器總會被執行的。起到了總攔截器的作用。
如果是REST風格的URL,靜態資源也會被攔截。
方案二, (近似) 總攔截器, 攔截匹配的URL。
Xml代碼
<mvc:interceptors > <mvc:interceptor> <mvc:mapping path="/user/*" /> <!-- /user/* --> <bean class="com.mvc.MyInteceptor"></bean> </mvc:interceptor> </mvc:interceptors>
就是比 方案一多了一個URL匹配。
如果是REST風格的URL,靜態資源也會被攔截。
方案三,HandlerMappint上的攔截器。
如果是REST風格的URL,靜態資源就不會被攔截。因為我們精準的注入了攔截器。
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 />,
它會自動注冊DefaultAnnotationHandlerMapping 與AnnotationMethodHandlerAdapter 這兩個bean,所以就沒有機會再給它注入interceptors屬性,就無法指定攔截器。
當然我們可以通過人工配置上面的兩個Bean,不使用 <mvc:annotation-driven />,就可以 給interceptors屬性 注入攔截器了。
其實我也不建議使用 <mvc:annotation-driven />,
而建議手動寫詳細的配置文件,來替代 <mvc:annotation-driven />
,這就控制力就強了。
如何替換 <mvc:annotation-driven />
?他到底做了什么工作?
一句 <mvc:annotation-driven />
實際做了以下工作:(不包括添加自己定義的攔截器)
我們了解這些之后,對Spring3 MVC的控制力就更強大了,想改哪就改哪里。
Xml代碼
<!-- 注解請求映射 --> <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"> <property name="interceptors"> <list> <ref bean="logNDCInteceptor"/> <!-- 日志攔截器,這是你自定義的攔截器 --> <ref bean="myRequestHelperInteceptor"/> <!-- RequestHelper攔截器,這是你自定義的攔截器--> <ref bean="myPermissionsInteceptor"/> <!-- 權限攔截器,這是你自定義的攔截器--> <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-->
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。