程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> 關於JAVA >> 如何在基於注解風格的Spring-MVC中使用攔截器

如何在基於注解風格的Spring-MVC中使用攔截器

編輯:關於JAVA

Spring-MVC如何使用攔截器,官方文檔只給出了非注解風格的例子。那麼基於注解風格如何使用攔截器呢?

基於注解基本上有2個可使用的定義類,分別是DefaultAnnotationHandlerMapping和AnnotationMethodHandlerAdapter:

<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"/>
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"/>

1、DefaultAnnotationHandlerMapping

DefaultAnnotationHandlerMapping本身支持自定義攔截器,只需按如下進行配置:

1 <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping">
2     <property name="interceptors">
3         <list>
4                <bean class="packageName.XXXInterceptor" />
5         </list>
6     </property>
7 </bean>

Interceptor的定義為:

 1 public class XXXInterceptor extends HandlerInterceptorAdapter {
2     @Override
3     public boolean preHandle(HttpServletRequest req, HttpServletResponse resp, Object handler) {
4
5         String className = handler.getClass().getName();// packageName.ClassName
6         if (Error) {
7             return false;
8         }
9         return true;
10     }
11 }

2、AnnotationMethodHandlerAdapter

目前,筆者沒找到如何給AnnotationMethodHandlerAdapter配置自定義Interceptor的方法,但是有個customArgumentResolver可以利用一下,來充當Interceptor。

1 <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
2     <property name="customArgumentResolver">
3         <bean class="packageName.XXXResolver"/>
4     </property>
5 </bean>

Resolver的定義為:

 1 public class XXXResolver implements WebArgumentResolver {
2
3     @Override
4     public Object resolveArgument(MethodParameter methodParameter, NativeWebRequest webRequest) throws Exception {
5
6         String className = methodParameter.getMethod().getDeclaringClass().getName();// packageName.ClassName
7
8         // 如何取得Response和Request
9         HttpServletResponse resp = (HttpServletResponse) webRequest.getNativeResponse();
10        HttpServletRequest req = (HttpServletRequest) webRequest.getNativeRequest();
11
12        if (Error) {
13            if (!resp.isCommitted()) resp.sendError(ERROR_STATUS);
14        }
15        return UNRESOLVED;
16     }
17 }
18

仔細的人會看出,第二種方法其實根本不是攔截。其實第二種只是在映射Controller,調用方法的時候,給每一個方法的參數增加了一個切點。

上例在出錯的時候往HttpServletResponse寫錯誤狀態,來通知web容器進行錯誤重定向,達到了攔截器的作用。

這麼做有一個缺點,就是每個參數都有自己的切點,比如方法有3個參數就會調3次resolveArgument。為了避免出錯,需要判斷一下resp.isCommitted。

customArgumentResolver的初衷不是用來做Interceptor的,但有些環境卻不得不使用它,比如部署在GAE上。

GAE是不支持DefaultAnnotationHandlerMapping的,因為此類用到了org.springframework.beans.BeanUtils.findEditorByConvention,這個方法會調用java.lang.ClassLoader.getSystemClassLoader,而這正是GAE所不允許的。

PS:

文中提到軟件的版本:

spring - 2.5.X

google app engine - 1.2.5

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved