一般情况下,对来自浏览器的请求的拦截,是利用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的使用相关推荐

  1. spring boot项目 中止运行 最常用的几种方法

    spring boot项目 中止运行 最常用的几种方法: 1. 调用接口,停止应用上下文 @RestController public class ShutdownController impleme ...

  2. html+spring boot简单的ajax数据传输实现

    本篇讲解在前后端不分离情况下的html+spring boot的项目数据传输实现 首先,后台我写了三个接口 package com.demo.ajax.controller;import com.de ...

  3. Spring Boot整合Spring Data JPA操作数据

    一. Sping Data JPA 简介 Spring Data JPA 是 Spring 基于 ORM 框架.JPA 规范的基础上封装的一套 JPA 应用框架,底层使用了 Hibernate 的 J ...

  4. Spring AOP + Redis解决重复提交的问题

    Spring AOP + Redis解决重复提交的问题 用户在点击操作的时候,可能会连续点击多次,虽然前端可以通过设置按钮的disable的属性来控制按钮不可连续点击,但是如果别人拿到请求进行模拟,依 ...

  5. Spring cloud 微服务docker容器化最佳实践

    Spring cloud 是当下最炙手可热的微服务套件,我们将介绍如何整合Docker容器达到高效快捷的构建发布 采用了dockerfile-maven-plugin插件发布镜像到远程docker主机 ...

  6. ssh(Struts+spring+Hibernate)三大框架整合-简述

    ssh(Struts+spring+Hibernate)三大框架配合使用来开发项目,是目前javaee最流行的开发方式,必须掌握: 注意: 为了稳健起见,每加入一个框架,我们就需要测试一下,必须通过才 ...

  7. Strutsw2与Spring整合流程-简述

    1.      新建WEB工程: 2.      导入struts2开发包,和资源配置文件 ① globalMessages.properties ② struts.properties 3.     ...

  8. spring Bean自动装配

    spring Bean自动装配 自动装配是使用spring满足bean依赖的一种方式. spring会在应用上下文中为某个bean寻找其依赖的bean. spring自动装配需要从两个角度来实现,或者 ...

  9. spring IOC创建对象方式

    spring IOC创建对象方式 通过无参构造来创建 验证方法: 创建实体类: public class User {private String name;public User() {System ...

最新文章

  1. 每个Web开发人员应该知道的12个终端命令
  2. C#中as与is的用法
  3. rust房子 如何拆除_小产权房“依法拆除”?中央新文件透露“1个新信号”
  4. CososJS学习笔记(1) 环境配置(填坑版,让你少走弯路!)
  5. linux系统挂载(卸载)U盘(文件系统)
  6. 正则表达式匹配C++代码实现
  7. Python包的相对导入时出现问题解决
  8. LInux下centos6.7 设置字符集,解决乱码问题
  9. 【共生性】CVPR 2019:Multi-Label Image Recognition with Graph Convolutional Networks
  10. 剑指offer之链表中环的入口结点
  11. 1.2 控制器 view 的创建和加载
  12. SPSS Modeler 报错
  13. 安泰测试新手教程-泰克数字示波器使用方法
  14. 基于遗传算法(deap库)的一元函数寻优代码详解
  15. 现场总线课程设计——IIC总线
  16. 隐藏桌面图标通过命令行启动
  17. 市场调研-全球与中国汽车零部件涂层市场现状及未来发展趋势
  18. VMware无法启动虚拟机里面没网络解决方法
  19. python农历_用python计算农历/阴历假日
  20. Redis集群搭建失败 connected_slaves:0

热门文章

  1. Unit Testing for WinForm
  2. AVL树C++实现(插入,删除,查找,清空,遍历操作)
  3. C# MBG 扩展方法类库 分享
  4. 《Groovy官方指南》翻译邀请
  5. mysql数据库基本操作练习
  6. DBMS_STATS.GATHER_TABLE_STATS详解
  7. memcached的基本命令(安装、卸载、启动、配置相关)
  8. 备忘录(scanf和continue)
  9. ubuntu 为用户添加组
  10. legend3---laravel中获取控制器名称和方法名称