Spring MVC 拦截器与过滤器的区别及执行顺序

一、拦截器(Interceptor) vs 过滤器(Filter)

特性 拦截器(Interceptor) 过滤器(Filter)
作用范围 Spring MVC 框架层面 Servlet 容器层面
依赖框架 需 Spring 容器支持 原生 Servlet 规范
访问对象 可访问 HandlerMethod、ModelAndView 等对象 仅能访问 ServletRequest/Response
业务关联 可获取 Spring Bean 和业务数据 无法直接访问 Spring 上下文
触发时机 在 Controller 方法前后执行 在请求进入 Servlet 前/响应返回后执行
异常处理 可通过 @ControllerAdvice 统一处理 需在 web.xml 配置错误页面

二、执行顺序

flowchart TD A[请求] --> B(Filter#doFilter) B --> C(Interceptor#preHandle) C --> D[Controller] D --> E(Interceptor#postHandle) E --> F(Interceptor#afterCompletion) F --> G(Filter#doFilter后续逻辑) G --> H[响应]
  1. 过滤器链(FilterChain)web.xml声明顺序依次执行或者按过滤器名称
  2. 拦截器链(Interceptor) 按注册顺序执行 preHandle(先进先出) 逆序执行 postHandleafterCompletion(后进先出)

三、案例:定义多个拦截器(完整配置示例)

java 复制代码
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration
public class InterceptorConfig implements WebMvcConfigurer {

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        // 1. 日志拦截器(最先执行)
        registry.addInterceptor(new LogInterceptor())
                .addPathPatterns("/**")
                .order(1); // 数字越小优先级越高

        // 2. 认证拦截器
        registry.addInterceptor(new AuthInterceptor())
                .addPathPatterns("/secure/**")
                .excludePathPatterns("/secure/public")
                .order(2);

        // 3. 性能监控拦截器(最后执行)
        registry.addInterceptor(new PerformanceInterceptor())
                .addPathPatterns("/api/**")
                .order(3);
    }
}

// 示例拦截器实现
class LogInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
        System.out.println("[Log] Request URI: " + request.getRequestURI());
        return true;
    }
}

class AuthInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
        if (request.getSession().getAttribute("user") == null) {
            response.setStatus(401);
            return false;
        }
        return true;
    }
}

class PerformanceInterceptor implements HandlerInterceptor {
    private ThreadLocal<Long> startTime = new ThreadLocal<>();
    
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
        startTime.set(System.currentTimeMillis());
        return true;
    }
    
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
        System.out.printf("[Performance] %s took %d ms%n", 
            request.getRequestURI(), 
            System.currentTimeMillis() - startTime.get()
        );
    }
}
3.1 关键配置说明
  1. 执行顺序控制 order()方法指定优先级(值越小越先执行)
  2. 路径匹配规则 addPathPatterns():设置需拦截的路径(支持 Ant 风格) excludePathPatterns():设置排除路径
  3. 拦截器生命周期 preHandle:Controller 方法执行前 postHandle:Controller 执行后,视图渲染前 afterCompletion:请求完成后(总会执行)

💡 最佳实践:将公共拦截器(如日志、跨域)设为高优先级,业务拦截器(如权限)设为低优先级。

相关推荐
hweiyu006 小时前
Go Fiber 简介
开发语言·后端·golang
你的人类朋友8 小时前
😎 Node.js 应用多阶段构建 Dockerfile 详解
后端·docker·容器
小坏讲微服务8 小时前
Spring Boot整合Redis注解,实战Redis注解使用
spring boot·redis·分布式·后端·spring cloud·微服务·mybatis
橘子海全栈攻城狮8 小时前
【源码+文档+调试讲解】基于Spring Boot的考务管理系统设计与实现 085
java·spring boot·后端·spring
追逐时光者9 小时前
一个基于 .NET 8 + DDD 搭建的模块化微服务框架
后端·.net
William_cl9 小时前
C# ASP.NET MVC 数据验证实战:View 层双保险(Html.ValidationMessageFor + jQuery Validate)
后端·c#·asp.net·mvc
Access开发易登软件9 小时前
Access导出带图表的 HTML 报表:技术实现详解
数据库·后端·html·vba·导出·access
Archy_Wang_19 小时前
ASP.NET Core 应用的零停机部署策略
后端·servlet·asp.net
无责任此方_修行中10 小时前
一行代码的“法律陷阱”:开发者必须了解的开源许可证知识
前端·后端·开源
合作小小程序员小小店10 小时前
web网页开发,在线物流管理系统,基于Idea,html,css,jQuery,jsp,java,SSM,mysql
java·前端·后端·spring·intellij-idea·web