Atea - Hero's Grave

面向对象,开源,框架,敏捷,云计算,NoSQL,商业智能,编程思想。

  BlogJava :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 管理 ::
  40 随笔 :: 0 文章 :: 28 评论 :: 0 Trackbacks
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

参考资料
http://www.scottmurphy.info/spring_framework_annotation_based_controller_interceptors
http://groups.google.com/group/google-appengine-java/browse_thread/thread/54aadf8164be7bf0
posted on 2009-09-20 20:23 Atea 阅读(7255) 评论(0)  编辑  收藏 所属分类: SpringGAE

只有注册用户登录后才能发表评论。


网站导航: