MVC的攔截器
經本人在Spring mvc中對方案1和方案2的測試表明,並沒有攔截靜態資源,所以可以放心使用方案1和方案2,方案3可以放棄,並且可以放心使用<mvc:annotation-driven />注解。
方案一,(近似)總攔截器,攔截所有url
<mvc:interceptors> <bean class="com.app.mvc.MyInteceptor" /> </mvc:interceptors>
為什么叫“近似”,前面說了,Spring沒有總的攔截器。
<mvc:interceptors/>會為每一個HandlerMapping,注入一個攔截器。總有一個HandlerMapping是可以找到處理器的,最多也只找到一個處理器,所以這個攔截器總會被執行的。起到了總攔截器的作用。
如果是REST風格的URL,靜態資源也會被攔截。(在4.0上測試並未有此問題)
方案二, (近似) 總攔截器, 攔截匹配的URL。
<mvc:interceptors > <mvc:interceptor> <mvc:mapping path="/user/*" /> <!-- /user/* --> <bean class="com.mvc.MyInteceptor"></bean> </mvc:interceptor> </mvc:interceptors>
就是比 方案一多了一個URL匹配。
如果是REST風格的URL,靜態資源也會被攔截。(在4.0上測試並未有此問題)
方案三,HandlerMappint上的攔截器。
如果是REST風格的URL,靜態資源就不會被攔截。因為我們精准的注入了攔截器
<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 />實際做了以下工作:(不包括添加自己定義的攔截器)
我們了解這些之后,對Spring3 MVC的控制力就更強大了,想改哪就改哪里。
spring 3.0.x是下面的配置
<!-- 注解請求映射 --> <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"> <property name="interceptors"> <list> <ref bean="logNDCInteceptor"/> <!-- 日志攔截器,這是你自定義的攔截器 --> </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-->
轉載:http://elf8848.iteye.com/blog/875830
spring 3.1 later:
Spring 3.0.x中使用了annotation-driven后,缺省使用DefaultAnnotationHandlerMapping 來注冊handler method和request的mapping關系。
AnnotationMethodHandlerAdapter來在實際調用handlermethod前對其參數進行處理。
並在dispatcherServlet中,當用戶未注冊自定義的ExceptionResolver時,注冊AnnotationMethodHandlerExceptionResolver來對使用@ExceptionHandler標注的異常處理函數進行解析處理(這也導致當用戶注冊了自定義的exeptionResolver時將可能導致無法處理@ExceptionHandler)。
在spring mvc 3.1中,對應變更為
DefaultAnnotationHandlerMapping -> org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping
AnnotationMethodHandlerAdapter -> org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter
AnnotationMethodHandlerExceptionResolver -> ExceptionHandlerExceptionResolver
以上都在使用了annotation-driven后自動注冊。
而且對應分別提供了AbstractHandlerMethodMapping , AbstractHandlerMethodAdapter和 AbstractHandlerMethodExceptionResolver以便於讓用戶更方便的實現自定義的實現類
<mvc:annotation-driven />的可選配置
<mvc:annotation-driven message-codes-resolver ="bean ref" validator="" conversion-service=""> <mvc:return-value-handlers> <bean></bean> </mvc:return-value-handlers> <mvc:argument-resolvers> </mvc:argument-resolvers> <mvc:message-converters> </mvc:message-converters>[/color] </mvc:annotation-driven>
具體可以參見:http://starscream.iteye.com/blog/1098880
Spring攔截器的定義:
Spring為我們提供了:
org.springframework.web.servlet.HandlerInterceptor接口,
org.springframework.web.servlet.handler.HandlerInterceptorAdapter適配器,
實現這個接口或繼承此類,可以非常方便的實現自己的攔截器。
有以下三個方法:
//Action之前執行: public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler); //如果返回false則中斷請求 //生成視圖之前執行 public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView); //最后執行,可用於釋放資源 public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)