过滤器、拦截器、ControllerAdvice和AOP

本文介绍了过滤器、拦截器、ControllerAdvice和AOP在Spring中的应用场景、执行顺序及原理。内容包括过滤器的链式调用、拦截器的调用流程、ControllerAdvice的异常处理以及AOP的代理方式。

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

面试官:说说过滤器和拦截器的区别? 这个问题面试题库算是比较经典的,这两个我相信很多同学在工作中都有接触过,但如果没有经过系统的整理,还真的不好说出个123来,那老湿机在此这面就把它俩和常用的AOP、ControllerAdvice放一起,带你做一个比较全面的认识。

1. 了解4种拦截方法的执行顺序

先上一个栗子,看四种拦截方法并驾齐驱使用时,谁先谁后:

/**
 * ============Filter过滤器============
 */
@Slf4j
@WebFilter(urlPatterns = "/*")
public class DemoFilter implements Filter {
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) throws IOException, ServletException {
        log.info("Filter 进入");
        filterChain.doFilter(request, response);
        log.info("Filter 退出");
    }
}
l
/**
 * ============Interceptor过滤器============
 */
public class DemoInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        log.info("Interceptor preHandle 进入");
        return true;
    }
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        log.info("Interceptor postHandle 进入");
    }
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        log.info("Interceptor afterCompletion 进入");
    }
}

/**
 * ============ControllerAdvice============
 */
@ControllerAdvice
public class DemoControllerAdvice {

    @InitBinder
    public void init(WebDataBinder binder) {
        log.info("ControllerAdvice init 进入");
        binder.setFieldDefaultPrefix("user.");
    }
    @ExceptionHandler(Exception.class)
    public String handleException(Exception e) {
        log.info("ControllerAdvice handleException 进入");
        return "error";
    }
}

/**
 * ============AOP============
 */
@Aspect
@Component
public class DemoAspect {
    @Pointcut("(@target(org.springframework.web.bind.annotation.RestController)) " +
            "&& within(cn.demo.api..*) && execution(public * *(..))")
    public void pointcut() {
    }

    @Around(value = "pointcut()")
    public Object around(ProceedingJoinPoint pjp) throws Throwable {
        log.info("Aop 进入");
        Object proceed = pjp.proceed();
        log.info("Aop 退出");
        return proceed;
    }
}

/**
 * ============控制器入口============
 */
@RestController
@RequestMapping("/demo")
public class DemoController {
    @RequestMapping("/test")
    public Map<String,
### 过滤器拦截器的执行顺序 在Web应用程序开发中,过滤器(Filter)拦截器(Interceptor)都是用来处理HTTP请求的重要组件。两者的主要区别在于它们的工作阶段以及如何配置其优先级。 #### 执行顺序概述 对于Spring MVC框架而言,当一个HTTP请求到达服务器时,最先触发的是过滤器[^1]。这些过滤器可以对进入系统的每一个请求应用通用逻辑,比如身份验证、日志记录等操作。接着才是拦截器发挥作用的时间点——即在`DispatcherServlet`已经接收到了请求之后但在实际业务处理器(Controller中的方法)被执行前的一段时间内[^2]。 具体来说: - **过滤器**:位于最前端,负责在整个请求/响应周期内的预处理工作; - **拦截器**:紧随其后,在具体的路由解析完成后但目标动作执行之前介入; 因此,标准的执行路径如下所示: 1. Filter过滤器 2. Interceptor拦截器 3. ControllerAdvice全局异常捕获或数据绑定支持等功能 4. AOP面向切面编程机制[^3] #### 优先级设定方式 为了控制多个同类型中间件之间的相对位置关系,可以通过以下几种手段来调整它们各自的加载次序: ##### 对于过滤器 实例化`Filter`类并注册至容器的同时需指明URL匹配模式、名称及排序权重参数。这决定了哪些类型的请求会被该过滤器所影响及其相对于其他同类插件的位置。例如通过实现`Ordered`接口来自定义排序数值越低则代表越早被调用。 ```java @Component public class MyCustomFilter implements Filter, Ordered { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { // 自定义逻辑... chain.doFilter(request,response); } @Override public int getOrder(){ return HIGHEST_PRECEDENCE; } } ``` ##### 针对拦截器 同样地,在向Spring上下文中注入自定义拦截器对象的时候也要考虑好它的出场时机。一般情况下可通过重写`addInterceptors()`方法并将所需实例加入列表的方式来完成这项任务。值得注意的是,这里还可以利用`PathMatcher`进一步细化适用范围从而间接影响整体排列结构。 ```java @Override protected void addInterceptors(InterceptorRegistry registry){ registry.addInterceptor(new CustomInterceptor()) .addPathPatterns("/api/**")// 设置作用域 .order(Ordered.HIGHEST_PRECEDENCE); // 设定最高优先级 } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值