【Java必学框架】一文搞懂Spring拦截器

本文介绍了SpringMVC拦截器的概念、快速入门,重点讲解了如何利用拦截器实现登录鉴权,并详细分析了拦截器链的工作原理。通过实例展示了拦截器在预处理、后处理和异常处理中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.概念

我们知道,j2ee中自带过滤器,其是web三大组件之一。多个过滤器形成的过滤链可以对请求进行一系列的加工处理,鉴权转发等预处理与后处理。

那么什么是拦截器呢?

Spring MVC 的拦截器类似于 Servlet 开发中的过滤器 Filter,用于对处理器进行预处理和后处理。 将拦截器按一定的顺序联结成一条链,这条链称为拦截器链(Interceptor Chain)。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用。拦截器也是AOP思想的具体实现。

以下是过滤器与拦截器的区别:

区别过滤器(Filter)拦截器(Interceptor)
使用范围是 servlet 规范中的一部分,任何 Java Web 工程都可以使用是 SpringMVC 框架自己的,只有使用了 SpringMVC 框架的工程才能用
拦截范围在 url-pattern 中配置了/*之后, 可以对所有要访问的资源拦截<mvc:mapping path=“”/>中配置了/**之 后,会对所有资源进行拦截。但是可以通过标签<mvc:exclude-mapping path=“”/>排除不需要拦截的资源

2.快速入门

步骤:

​ 1.创建拦截器类实现HandlerInterceptor接口

​ 2.在spring-mvc.xml文件中声明拦截器

以下进行演示:

HandlerInterceptor类里面包含的三个方法不是抽象方法,而是默认方法。我们可以按需选取进行重写。其中preHandle是请求处理之前进行调用,返回值为true表示放行。

public class LoginInterceptor implements HandlerInterceptor {

    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("preHandle...");
        return true;
    }

    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("postHandle...");
    }

    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("afterHandle...");
    }
}

配置拦截器

<!--声明拦截器-->
<mvc:interceptors>
    <mvc:interceptor>
        <mvc:mapping path="/**"/>
        <bean class="venture.study.interceptor.LoginInterceptor"></bean>
    </mvc:interceptor>
</mvc:interceptors>

测试类:

@RequestMapping("/test")
@ResponseBody
public void test(){
    System.out.println("拦截器测试...");
}

访问/test,控制台输出如下:

preHandle…
拦截器测试…
postHandle…
afterHandle…

根据以上输出结果,我们可以将三个方法被调用时期以及作用整理如下表:

方法说明
preHandler该方法在请求处理之前进行调用,该方法的返回值是布尔值Boolean类型的。返回值为false 时,表示请求结束,后续的Interceptor 和Controller 都不会再执行;返回值为true 时就会继续调用下一个Interceptor 的preHandle 方法,或者执行Controller1
postHandler该方法是在当前请求进行处理之后被调用,前提是preHandle 方法的返回值为 true 时才能被调用,且它会在DispatcherServlet 进行视图返回渲染之前被调用,所以我们可以在这个方法中对Controller 处理之后的ModelAndView 对象进行操作
afterCompletion该方法将在整个请求结束之后,也就是在DispatcherServlet 渲染了对应的视图之后执行,前提是preHandle 方法的返回值为true 时才能被调用

3.拦截器链

有多个拦截器的时候拦截器是如何工作的呢?

这里说明一下,拦截器的先后执行顺序与配置先后有关,先配置的拦截器先执行。访问/test,输出结果如下:

preHandle…
第二个preHandle…
拦截器测试…
第二个postHandle…
postHandle…
第二个afterHandle…
afterHandle…

分析结果,我们可以看到先配置的preHandle方法先执行,但是postHandle与afterHandle方法却后执行,后进先出,跟队列差不多。

4.利用拦截器实现登录鉴权

登录控制器:

@RequestMapping("/user-login")
public String userLogin(String username, String password, HttpServletRequest request){
    User user = userService.login(username,password);
    if(user != null){
        request.getSession().setAttribute("user",user);//存session
        return "redirect:/user/user-list";
    }else{
        return "redirect:/login.jsp";
    }
}

鉴权拦截器,只需处理preHandle方法

public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    User user = (User)request.getSession().getAttribute("user");
    if(user != null){
        return true;//登陆了则放行
    }else{
        response.sendRedirect( request.getContextPath() +  "/login.jsp");//没有登陆则跳转到登录界面
        return false;
    }
}

配置拦截器,这里需要对登录控制器user-login进行放行

<mvc:interceptors>
    <mvc:interceptor>
        <mvc:mapping path="/**"/>
        <mvc:exclude-mapping path="/user/user-login"></mvc:exclude-mapping>
        <bean class="venture.study.interceptor.LoginInterceptor"></bean>
    </mvc:interceptor>
</mvc:interceptors>

  1. 如果有还有下一个拦截器,就接着调用下一个interceptor的preHandler()方法,否则就执行Controller ↩︎

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值