基于注解风格的Spring-MVC的拦截器

来源:互联网 发布:淘宝美工和客服哪个好 编辑:程序博客网 时间:2024/06/07 09:27

基于注解风格的Spring-MVC的拦截器

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(); // package Name .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所不允许的。

0 0
原创粉丝点击