拦截webapp下jsp_Spring MVC中的拦截器/过滤器HandlerInterceptorAdapter的使用

本文介绍了Spring MVC中如何使用HandlerInterceptorAdapter实现请求拦截,包括preHandle、postHandle和afterCompletion三个关键方法的作用,并通过示例展示了如何配置和使用拦截器进行URL映射和权限控制。同时,强调了在XML和注解配置下设置拦截器的区别。
摘要由CSDN通过智能技术生成

I分享永久IDEA激活码:ntelliJ IDEA 2020.1.2 激活码,永久有效(点击该链接)!

00e6dc2e9a55e78e156e587ed1bc6aab.png

一般情况下,对来自浏览器的请求的拦截,是利用Filter实现的

而在Spring中,基于Filter这种方式可以实现Bean预处理、后处理。 比如注入FilterRegistrationBean,然后在这个Bean上传递自己继承Filter实现的自定义Filter进入即可。

而Spring MVC也有拦截器,不仅可实现Filter的所有功能,还可以更精确的控制拦截精度。 

Spring MVC提供的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 {      }

preHandle在业务处理器处理请求之前被调用。预处理,可以进行编码、安全控制等处理; 

postHandle在业务处理器处理请求执行完成后,生成视图之前执行。后处理(调用了Service并返回ModelAndView,但未进行页面渲染),有机会修改ModelAndView; 

afterCompletion在DispatcherServlet完全处理完请求后被调用,可用于清理资源等。返回处理(已经渲染了页面),可以根据ex是否为null判断是否发生了异常,进行日志记录;

如果基于XML配置使用Spring MVC,可以利用SimpleUrlHandlerMapping、BeanNameUrlHandlerMapping进行Url映射(相当于struts的path映射)和拦截请求(注入interceptors)。

如果基于注解使用Spring MVC,可以使用DefaultAnnotationHandlerMapping注入interceptors。

注意无论基于XML还是基于注解,HandlerMapping Bean都是需要在XML中配置的。 

示例一:

在这个例子中,我们假设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;       }@Overridepublic 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(nowclosingTime){                     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{@Autowiredprivate UserService userService;@RequestMapping(params="action=reg")public ModelAndView reg(Users user) throws Exception {       userService.addUser(user);       return new ModelAndView("profile","user",user); }// other option ...

也可以配置多个拦截器,每个拦截器进行不同的分工。

示例二:

主要是XML配置不一样

  <mvc:interceptors>        <mvc:interceptor>          <mvc:mapping path="/" />      <mvc:mapping path="/user/**" />      <mvc:mapping path="/test/**" />        <bean class="com.alibaba.interceptor.CommonInterceptor">bean>        mvc:interceptor>    mvc:interceptors> 
package com.alibaba.interceptor;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.web.servlet.ModelAndView;import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;import com.alibaba.util.RequestUtil;public class CommonInterceptor extends HandlerInterceptorAdapter{       private final Logger log = LoggerFactory.getLogger(CommonInterceptor.class);       public static final String LAST_PAGE = "com.alibaba.lastPage";/** 利用正则映射到需要拦截的路径private String mappingURL;public void setMappingURL(String mappingURL) {this.mappingURL = mappingURL;}*//*** 在业务处理器处理请求之前被调用* 如果返回false*     从当前的拦截器往回执行所有拦截器的afterCompletion(),再退出拦截器链* 如果返回true*    执行下一个拦截器,直到所有的拦截器都执行完毕*    再执行被拦截的Controller*    然后进入拦截器链,*    从最后一个拦截器往回执行所有的postHandle()*    接着再从最后一个拦截器往回执行所有的afterCompletion()*/@Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object handler) throws Exception {      if ("GET".equalsIgnoreCase(request.getMethod())) {           RequestUtil.saveRequest();       }      log.info("==============执行顺序: 1、preHandle================");      String requestUri = request.getRequestURI();      String contextPath = request.getContextPath();      String url = requestUri.substring(contextPath.length());      log.info("requestUri:"+requestUri);      log.info("contextPath:"+contextPath);      log.info("url:"+url);      String username =  (String)request.getSession().getAttribute("user");      if(username == null){        log.info("Interceptor:跳转到login页面!");        request.getRequestDispatcher("/WEB-INF/jsp/login.jsp").forward(request, response);        return false;        }else          return true;}/*** 在业务处理器处理请求执行完成后,生成视图之前执行的动作* 可在modelAndView中加入数据,比如当前时间*/@Overridepublic void postHandle(HttpServletRequest request,HttpServletResponse response, Object handler,ModelAndView modelAndView) throws Exception {       log.info("==============执行顺序: 2、postHandle================");       if(modelAndView != null){  //加入当前时间          modelAndView.addObject("var", "测试postHandle");       }}/*** 在DispatcherServlet完全处理完请求后被调用,可用于清理资源等** 当有拦截器抛出异常时,会从当前拦截器往回执行所有的拦截器的afterCompletion()*/@Overridepublic void afterCompletion(HttpServletRequest request,HttpServletResponse response, Object handler, Exception ex)throws Exception {       log.info("==============执行顺序: 3、afterCompletion================");       }}
61125af4ff79cdcd6cf1e0dd9548b126.gif

如有文章对你有帮助,

“在看”和转发是对我大的支持

78afb4d603c31a45560cc3663d99736e.gif

关注白码啸西风

每天学习Java技术

3c43b8a5c975cd764efe1add88cf6656.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值