Spring為我們提供了:
org.springframework.web.servlet.HandlerInterceptor接口,
org.springframework.web.servlet.handler.HandlerInterceptorAdapter適配器,
實現這個接口或繼承此類,可以非常方便的實現自己的攔截器。
有以下三個方法:
Action之前執行:
1
2
|
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler); |
生成視圖之前執行
1
2
3
|
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView); |
最后執行,可用于釋放資源
1
2
|
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) |
分別實現預處理、后處理(調用了Service并返回ModelAndView,但未進行頁面渲染)、返回處理(已經渲染了頁面)
在preHandle中,可以進行編碼、安全控制等處理;
在postHandle中,有機會修改ModelAndView;
在afterCompletion中,可以根據ex是否為null判斷是否發生了異常,進行日志記錄。
參數中的Object handler是下一個攔截器。
如何使用攔截器?
自定義一個攔截器,要實現HandlerInterceptor接口:
Java代碼
1
2
3
|
public class MyInteceptor implements HandlerInterceptor { 略。。。 } |
Spring MVC并沒有總的攔截器,不能對所有的請求進行前后攔截。
Spring MVC的攔截器,是屬于HandlerMapping級別的,可以有多個HandlerMapping ,每個HandlerMapping可以有自己的攔截器。
當一個請求按Order值從小到大,順序執行HandlerMapping接口的實現類時,哪一個先有返回,那就可以結束了,后面的HandlerMapping就不走了,本道工序就完成了。就轉到下一道工序了。
攔截器會在什么時候執行呢? 一個請求交給一個HandlerMapping時,這個HandlerMapping先找有沒有處理器來處理這個請求,如何找到了,就執行攔截器,執行完攔截后,交給目標處理器。
如果沒有找到處理器,那么這個攔截器就不會被執行。
在spring MVC的配置文件中配置有三種方法:
方案一,(近似)總攔截器,攔截所有url
Java代碼
1
2
3
|
<mvc:interceptors> <bean class = "com.app.mvc.MyInteceptor" /> </mvc:interceptors> |
為什么叫“近似”,前面說了,Spring沒有總的攔截器。
<mvc:interceptors/>
會為每一個HandlerMapping,注入一個攔截器。總有一個HandlerMapping是可以找到處理器的,最多也只找到一個處理器,所以這個攔截器總會被執行的。起到了總攔截器的作用。
如果是REST風格的URL,靜態資源也會被攔截。
方案二, (近似) 總攔截器, 攔截匹配的URL。
Xml代碼
1
2
3
4
5
6
|
<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代碼
1
2
3
4
5
6
7
|
<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代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
<!-- 注解請求映射 --> <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--> |