• 基于注解风格的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所不允许的。
    PS:
    文中提到软件的版本:
    spring - 2.5.X

  • 相关阅读:
    通过IDEA解决spring配置文件
    idea中xml打开方式变成file,改回来
    idea 搜索不到前端的ajax controller接口的原因
    IDEA 出现 updating indices 卡进度条问题的解决方案并加快索引速度
    java 循环中使用list时,出现list中全部加入了对象导致没有实现分组的解决方案
    java 从字符串中 以单个或多个空格进行分隔 提取字符串
    idea ssm项目出现日志中文乱码,封装的json中的msg字段中文乱码(但是json封装的bean中的字段不乱码)等其他各种项目下的中文乱码解决方案
    javaweb 解决jsp中${}传递中文值到后端以及get请求中文乱码的问题
    idea 开发javaee 时,出现访问的文件和源文件不一样,没有正常更新的解决方案
    java 迭代器只遍历了一次的解决方案
  • 原文地址:https://www.cnblogs.com/suizhikuo/p/3440444.html
Copyright © 2020-2023  润新知