Spring MVC_HandlerInterceptorAdapter的使用

转载 2014年03月24日 22:29:54

一般情况下,对来自浏览器的请求的拦截,是利用Filter实现的,这种方式可以实现Bean预处理、后处理。 
Spring MVC的拦截器不仅可实现Filter的所有功能,还可以更精确的控制拦截精度。 
Spring为我们提供了org.springframework.web.servlet.handler.HandlerInterceptorAdapter这个适配器,继承此类,可以非常方便的实现自己的拦截器。他有三个方法:

public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)     
        throws Exception {     
        return true;    

    }     
    public void postHandle(     
            HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView)     
            throws Exception {     
    }     
    public void afterCompletion(     
            HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)     
            throws Exception {     
    } 
分别实现预处理、后处理(调用了Service并返回ModelAndView,但未进行页面渲染)、返回处理(已经渲染了页面)

 发起请求,进入拦截器链,运行所有拦截器的preHandle方法,
1.当preHandle方法返回false时,从当前拦截器往回执行所有拦截器的afterCompletion方法,再退出拦截器链。
2.当preHandle方法全为true时,执行下一个拦截器,直到所有拦截器执行完。再运行被拦截的Controller。然后进入拦截器链,运行所有拦截器的postHandle方法,完后从最后一个拦截器往回执行所有拦截器的afterCompletion方法.
当有拦截器抛出异常时,会从当前拦截器往回执行所有拦截器的afterCompletion方法


preHandle方法:返回true,映射处理器执行链将继续执行;当返回false时,DispatcherServlet处理器认为拦截器已经处理完了请求,而不继续执行执行链中的其它拦截器和处理器。
它的API文档解释如下:true if the execution chain should proceed with the next interceptor or the handler itself. Else, DispatcherServlet assumes that this interceptor has already dealt with the response itself.


在preHandle中,可以进行编码、安全控制等处理; 
在postHandle中,有机会修改ModelAndView; 
在afterCompletion中,可以根据ex是否为null判断是否发生了异常,进行日志记录。 

如果基于xml配置使用Spring MVC, 
可以利用SimpleUrlHandlerMapping、BeanNameUrlHandlerMapping进行Url映射(相当于struts的path映射)和拦截请求(注入interceptors), 
如果基于注解使用Spring MVC,可以使用DefaultAnnotationHandlerMapping注入interceptors。 
注意无论基于xml还是基于注解,HandlerMapping bean都是需要在xml中配置的。



一个demo: 
在这个例子中,我们假设UserController中的注册操作只在9:00-12:00开放,那么就可以使用拦截器实现这个功能。

 

public class TimeBasedAccessInterceptor extends HandlerInterceptorAdapter {     
    private int openingTime;     
    private int closingTime;     
    private String mappingURL;//利用正则映射到需要拦截的路径     
    public void setOpeningTime(int openingTime) {     
        this.openingTime = openingTime;     
    }     
    public void setClosingTime(int closingTime) {     
        this.closingTime = closingTime;     
    }     
    public void setMappingURL(String mappingURL) {     
        this.mappingURL = mappingURL;     
    }     
    @Override     
    public boolean preHandle(HttpServletRequest request,     
            HttpServletResponse response, Object handler) throws Exception {     
        String url=request.getRequestURL().toString();     
        if(mappingURL==null || url.matches(mappingURL)){     
            Calendar c=Calendar.getInstance();     
            c.setTime(new Date());     
            int now=c.get(Calendar.HOUR_OF_DAY);     
            if(now<openingTime || now>closingTime){     
                request.setAttribute("msg", "注册开放时间:9:00-12:00");     
                request.getRequestDispatcher("/msg.jsp").forward(request, response);     
                return false;     
            }     
            return true;     
        }     
        return true;     
    }     
}    

xml配置:

<bean id="timeBasedAccessInterceptor" class="com.spring.handler.TimeBasedAccessInterceptor">     
    <property name="openingTime" value="9" />     
    <property name="closingTime" value="12" />     
    <property name="mappingURL" value=".*/user\.do\?action=reg.*" />     
</bean>     
<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping">     
    <property name="interceptors">     
        <list>     
            <ref bean="timeBasedAccessInterceptor"/>     
        </list>     
    </property>     
</bean> 


这里我们定义了一个mappingURL属性,实现利用正则表达式对url进行匹配,从而更细粒度的进行拦截。当然如果不定义mappingURL,则默认拦截所有对Controller的请求。


UserController:
@Controller     
@RequestMapping("/user.do")     
public class UserController{     
    @Autowired     
    private UserService userService;     
    @RequestMapping(params="action=reg")     
    public ModelAndView reg(Users user) throws Exception {     
        userService.addUser(user);     
        return new ModelAndView("profile","user",user);     
    }     
    // other option ...     
}    
这个Controller相当于Struts的DispatchAction 
你也可以配置多个拦截器,每个拦截器进行不同的分工.


Spring MVC 中 HandlerInterceptorAdapter的使用

一般情况下,对来自浏览器的请求的拦截,是利用Filter实现的,这种方式可以实现Bean预处理、后处理。 Spring MVC的拦截器不仅可实现Filter的所有功能,还可以更精确的控制拦截精度...

SpringMVC拦截器(包括自定以拦截器--实现HandlerInterceptorAdapter)(资源和权限管理)

SpringMVC拦截器(资源和权限管理) 分类: 开源框架和工具 springmvc拦截器权限管理Interceptor访问控制 1.DispatcherServlet...

SpringMVC中使用Interceptor拦截器顺序等

SpringMVC 中的Interceptor 拦截器也是相当重要和相当有用的,它的主要作用是拦截用户的请求并进行相应的处理。比如通过它来进行权限验证,或者是来判断用户是否登陆,或者是像12306 那...

web拦截器中读取request中流导致跳转controller失败问题解决方案

请求的URL为:http://localhost:8980/helper/API/toQrcLogin web.xml中添加过滤器 sessionFilter com.Long.api....

HandlerInterceptorAdapter实现简单拦截

以下是本人开发中使用的简单实现,望各位大神提出见解及改进方法,也写此博客记录 一、继承org.springframework.web.servlet.handler.HandlerIntercept...

SpringMVC HandlerInterceptorAdapter登陆验证拦截器

SpringMVC HandlerInterceptorAdapter拦截器,方便,实用性大。 Spring MVC的拦截器不仅可实现Filter的所有功能,还可以更精确的控制拦截精度。...

springMVC:HandlerInterceptor拦截器的使用

1.使用背景Web项目中需要判断http接口用户Post上来的数据是否合法,如果不合法要另做处理,用户Post上来的数据是Json形式的,我们用了@RequestBody标记自动将json形式的提交封...
  • sunp823
  • sunp823
  • 2016年06月16日 21:08
  • 14440

Spring MVC中拦截器HandlerInterceptorAdapter中的preHandle方法

拦截器:顾名思义,就是对请求进行拦截,做一些预处理、后处理或返回处理的操作  Spring MVC中使用拦截器的方法,继承HandlerInterceptorAdapter类,并根据需求实现其中的pr...
  • zglwy
  • zglwy
  • 2017年01月24日 23:36
  • 2908

Spring HandlerInterceptor的使用

HandlerInterceptor翻译过来就是spring拦截器,它在某些功能应用上特别有用:1. 用户是否登陆以及用户权限管理 (见http://www.ideawu.net/ideablog/c...
  • totogogo
  • totogogo
  • 2007年05月25日 16:56
  • 25930

基于注解,HandlerInterceptorAdapter 拦截器

基于注解,HandlerInterceptorAdapter 拦截器第一步:自定义注解import java.lang.annotation.ElementType; import java.lang...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Spring MVC_HandlerInterceptorAdapter的使用
举报原因:
原因补充:

(最多只允许输入30个字)