基于注解风格的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
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值