zoukankan      html  css  js  c++  java
  • 基于注解风格的Spring-MVC的拦截器

    基于注解风格的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本身支持自定义拦截器,只需按如下进行配置:

    <bean
            class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping">
            <property name="interceptors">
                <list>
                    <bean class="com.sd.microMsg.interceptor.CharSetInterceptor"></bean>
                </list>
            </property>
        </bean>

    Interceptor的定义为:

    public class CharSetInterceptor extends HandlerInterceptorAdapter {
    
        @Override
        public boolean preHandle(HttpServletRequest request,
                HttpServletResponse response, Object handler) throws Exception {
            this.setBefor(request, response);
            return super.preHandle(request, response, handler);
        }
    
        /**
         * @throws UnsupportedEncodingException
         * 
         */
        private void setBefor(HttpServletRequest request,
                HttpServletResponse response) throws UnsupportedEncodingException {
            request.setCharacterEncoding("utf-8");
            response.setCharacterEncoding("utf-8");
            response.setContentType("text/html;charset=utf-8");
            System.out.println("utf-9");
    
        }
    }

    2、AnnotationMethodHandlerAdapter 
           目前,笔者没找到如何给AnnotationMethodHandlerAdapter配置自定义Interceptor的方法,但是有个customArgumentResolver可以利用一下,来充当Interceptor。

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

    Resolver的定义为:

    public class XXXResolver implements WebArgumentResolver {
    
        @Override
        public Object resolveArgument(MethodParameter methodParameter,
                NativeWebRequest webRequest) throws Exception {
    
            String className = methodParameter.getMethod().getDeclaringClass()
                    .getName(); // packageName.ClassName
    
            // 如何取得Response和Request
            HttpServletResponse resp = (HttpServletResponse) webRequest
                    .getNativeResponse();
            HttpServletRequest req = (HttpServletRequest) webRequest
                    .getNativeRequest();
    
            if (Error) {
                if (!resp.isCommitted())
                    resp.sendError(ERROR_STATUS);
            }
            return UNRESOLVED;
        }
    }

      仔细的人会看出,第二种方法其实根本不是拦截。其实第二种只是在映射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

    来自 http://www.cnblogs.com/siukun/archive/2011/12/06/2277647.html

    我用第一种满足了我的需求!

  • 相关阅读:
    java基本类型和包装类型的区别以及object的公共方法
    Scrapy学习
    centos系统python2.7更新到3.5
    requests和BeautifulSoup模块的使用
    基于角色的权限控制系统(role-based access control)
    Git的使用
    可插拔式后台管理系统(Django)
    Django admin site应用
    【算法】RMQ LCA 讲课杂记
    oh-my-zsh 安装和使用
  • 原文地址:https://www.cnblogs.com/mjorcen/p/3747875.html
Copyright © 2011-2022 走看看