springboot--拦截器和接口-ResponseBodyAdvice--WebMvcConfigurer-filter

本文介绍了一种日志拦截器的实现方法,能够自动记录API的出入参及执行时间,减轻程序员负担。同时,展示了如何使用ResponseBodyAdvice对响应结果进行统一处理,包括数据包装和性能监控。

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

org.springframework.web.servlet.config.annotation.WebMvcConfigurer的实现类和javax.servlet.Filter的实现类在springboot项目中那个先执行呢

在 Spring Boot 项目中,javax.servlet.Filter 的实现类先于 WebMvcConfigurer 配置的拦截器(Interceptor)执行。以下是详细说明:


执行顺序

  1. Filter(过滤器)

    • 属于 Servlet 规范 的组件,作用于 Servlet 容器层
    • 执行时机:在请求进入 DispatcherServlet 之前,以及响应返回给客户端之前。
    • 顺序控制:通过 @Order 注解或 FilterRegistrationBean 设置优先级。
  2. WebMvcConfigurer 配置的拦截器(Interceptor

    • 属于 Spring MVC 框架 的组件,作用于 Spring MVC 层
    • 执行时机:在 DispatcherServlet 处理请求的过程中(Controller 方法调用前后)。
    • 顺序控制:通过 @Order 注解或 addInterceptor 方法的调用顺序。

请求处理流程

客户端请求 → 
  Servlet 容器层(Filter 链) → 
    DispatcherServlet → 
      Spring MVC 拦截器链(preHandle → Controller → postHandle → afterCompletion) → 
        响应返回客户端(再次经过 Filter 链)

关键区别

组件层级执行阶段配置方式
FilterServlet 容器层全局请求处理(请求到达 DispatcherServlet 前/后)@Component + @OrderFilterRegistrationBean
InterceptorSpring MVC 框架层针对 Spring MVC 请求(Controller 方法前后)WebMvcConfigurer.addInterceptors() + @Order

代码示例

1. Filter 实现类
@Component
@Order(1) // 控制 Filter 执行顺序(数字越小优先级越高)
public class CustomFilter implements Filter {
    @Override
    public void doFilter(
        ServletRequest request, 
        ServletResponse response, 
        FilterChain chain
    ) throws IOException, ServletException {
        System.out.println("Filter 执行前");
        chain.doFilter(request, response);
        System.out.println("Filter 执行后");
    }
}
2. WebMvcConfigurer 配置的拦截器
@Configuration
public class WebConfig implements WebMvcConfigurer {
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new CustomInterceptor())
                .addPathPatterns("/**")
                .order(2); // 控制拦截器执行顺序
    }
}

// 拦截器类
public class CustomInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
        System.out.println("Interceptor preHandle");
        return true;
    }

    @Override
    public void postHandle(...) {
        System.out.println("Interceptor postHandle");
    }

    @Override
    public void afterCompletion(...) {
        System.out.println("Interceptor afterCompletion");
    }
}

输出结果

Filter 执行前    → Filter 先执行
Interceptor preHandle → 拦截器在 Filter 之后执行
Controller 方法处理...
Interceptor postHandle
Interceptor afterCompletion
Filter 执行后    → Filter 在响应返回前再次执行

总结

  • Filter 先于 Interceptor 执行:因 Filter 属于 Servlet 容器层,处理请求更早。
  • 执行范围不同
    • Filter 可处理所有请求(包括静态资源)。
    • Interceptor 仅处理 Spring MVC 路由的请求。
  • 用途差异
    • Filter 适合全局性操作(如日志、跨域、权限)。
    • Interceptor 适合与业务逻辑相关的操作(如认证、参数校验)。

包装HttpServletRequest

添加链接描述

问题

在这里插入图片描述

目标

1 你可以在自己的日志中看到自己接口的出参和入参,同时你不用在每个接口中写打印的逻辑,直接交给拦截器完成。
2 你可以任意包装你的出参,比如加上消耗的时间,添加一个唯一的uuid等等,同时这些东西都是由拦截器提供的功能。这些功能也可以做成配置化,比如你那些接口不需要这样的包装,在yml中配置一下,就可以了。

core class for this function

import org.springframework.web.servlet.HandlerInterceptor
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;

接口模样:

@RestController
public class TestController{
@RequestMapping(value = "/test")
public Object test(){
return "hello word";
}
}

每次接口请求都会输出param,路径 消耗的时间
在这里插入图片描述
接口正常输出就是hello word,可是包装后
在这里插入图片描述

copy thoese code for use

package wl.log.intercept.interceptor;

import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.method.HandlerMethod;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Objects;

/**
 * @program: log-intercepter
 * @description: 日志拦截器
 * @create: 2020-05-02 20:52
 * 描述:
 */
@Slf4j
public class LogIntercepter implements HandlerInterceptor {
// 妈的,每次写接口都要你添加日志,监控你的 出入,累不累,直接用拦截器,他不香吗?
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        if (handler instanceof HandlerMethod) {
            HandlerMethod h = (HandlerMethod) handler;
            log.info("method is {}, param is{} ,path is {}",h.getMethod().getName(), JSON.toJSON(request.getParameterMap()),request.getRequestURI());
            request.setAttribute("startTime",System.currentTimeMillis());
        }
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        Long startTime = (Long)request.getAttribute("startTime");
        //在info级别才打印
        if (Objects.nonNull(startTime)&&log.isInfoEnabled()) {
         log.info("the url use time is {}",System.currentTimeMillis()-startTime);
        }

    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {

    }
}

WebMvcConfigurer code

package wl.log.intercept.interceptor;

import org.springframework.boot.SpringBootConfiguration;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
import org.springframework.web.servlet.handler.MappedInterceptor;

/**
 * @program: log-intercepter
 * @create: 2020-05-02 21:12
 * 描述:
 */
@SpringBootConfiguration
public class CustomerWebMVCConfig implements WebMvcConfigurer {
    /**
     * @param registry
     */
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new LogIntercepter()).addPathPatterns("/**");
    }
}

web 响应拦截器

package wl.log.intercept.interceptor;

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.serializer.SerializerFeature;
import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;
import wl.log.intercept.entity.global.ResponseResult;

import java.util.Objects;

/**
 * @program: log-intercepter
 * @description:
 * @create: 2020-05-02 22:30
 */
@ControllerAdvice
public class ApiResultIntercepter implements ResponseBodyAdvice<Object> {
    /**
     *  判断哪些需要拦截
     */
    @Override
    public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
        return true;
    }

    @Override
    public Object beforeBodyWrite(Object body,
     MethodParameter returnType,
      MediaType selectedContentType, 
      Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request,
       ServerHttpResponse response) {
        //
        ResponseResult finalResult = null;
        Class bodyClass = body != null ? body.getClass() : null;
        // handle body  many conditions
        if(Objects.isNull(body)) {
            finalResult = ResponseResult.data(null).setCode(200);
        }else {
            finalResult = ResponseResult.data(body).setCode(200);
        }
         return String.class == bodyClass ? JSON.toJSONString(finalResult, SerializerFeature.WriteNullStringAsEmpty) : finalResult;
    }
}

demo已经写好,请拿去使用吧

https://download.youkuaiyun.com/download/wlwork66/12390425

WebMvcConfigurer

WebMvcConfigurerAdapter已经过时,在新版本2.x中被废弃,原因是springboot2.0以后,引用的是spring5.0,而spring5.0取消了WebMvcConfigurerAdapter ,现在都是直接实现WebMvcConfigure

/** 解决跨域问题 **/
public void addCorsMappings(CorsRegistry registry) ;
/** 添加拦截器 **/
void addInterceptors(InterceptorRegistry registry);
/** 这里配置视图解析器 **/
void configureViewResolvers(ViewResolverRegistry registry);
/** 配置内容裁决的一些选项 **/
void configureContentNegotiation(ContentNegotiationConfigurer configurer);
/** 视图跳转控制器 **/
void addViewControllers(ViewControllerRegistry registry);
/** 静态资源处理 **/
void addResourceHandlers(ResourceHandlerRegistry registry);
/** 默认静态资源处理器 **/
void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer);

HandlerInterceptor

SpringWebMVC的处理器拦截器,类似于Servlet开发中的过滤器Filter,用于处理器进行预处理和后处理。
使用的场景
  1、日志记录,可以记录请求信息的日志,
  以便进行信息监控、信息统计、计算PV(Page View)等等。
  2、权限检查:如登陆检测,进入处理器检测是否登陆,
  如果没有直接返回到登陆页面。
  3、性能监控:有时候系统在某段时间莫名其妙的慢,
  可以通过拦截器在进入处理器之前记录开始时间,在处理完后记录结束时间,从而得到该请求的处理时间(如果有反向代理,如apache可以自动记录);

  4、通用行为:读取cookie得到用户信息并将用户对象放入请求,
  从而方便后续流程使用,还有如提取Locale、Theme信息等,
  只要是多个处理器都需要的即可使用拦截器实现。
  5:原理
  本质也是AOP(面向切面编程),也就是说符合横切关注点的所有功能都可以放入拦截器实现。
 
public interface HandlerInterceptor {

   /**
     * 预处理回调方法,实现处理器的预处理(如检查登陆),
     * 第三个参数为响应的处理器,自定义Controller
     * 返回值:true表示继续流程(如调用下一个拦截器或处理器);
     *    *       false表示流程中断(如登录检查失败),
     * 不会继续调用其他的拦截器或处理器,
     * 此时我们需要通过response来产生响应;
   */
    boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
            throws Exception;
   /**
     * 后处理回调方法,实现处理器的后处理(但在渲染视图之前),
     * 此时我们可以通过modelAndView(模型和视图对象)
     * 对模型数据进行处理或对视图进行处理,
     * modelAndView也可能为null。
   */
    void postHandle(
            HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView)
            throws Exception;
   /**
    * 整个请求处理完毕回调方法,即在视图渲染完毕时回调,
    * 如性能监控中我们可以在此记录结束时间并输出消耗时间,
    * 还可以进行一些资源清理,类似于try-catch-finally中的finall
   * 但仅调用处理器执行链中preHandle返回true的拦截器的afterCompletion。
   */
    void afterCompletion(
            HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
            throws Exception;
}

click

how

ResponseBodyAdvice接口是在Controller执行return之后,在response返回给浏览器或者APP客户端之前,执行的对response的一些处理。可以实现对response数据的一些统一封装或者加密等操作。

@Override
    //判断是否要执行beforeBodyWrite方法,true为执行,false不执行
    //有些情况是不需要进行封装的,所以这个方法非常nice
    public boolean supports(MethodParameter returnType, Class converterType) {
        return false;
    }

    @Override
    //对response处理的执行方法
    public Object beforeBodyWrite(Object body, MethodParameter returnType,
            MediaType selectedContentType, Class selectedConverterType,
            ServerHttpRequest request, ServerHttpResponse response) {
        return null;
    }
eg ------ Controller类,返回参数为OutputObject(一个自定义的javaBean),我们要通过ResponseBodyAdvice,对该类的所有方法返回的OutputObject中的部分数据进行统一加密处理。
package com.cmos.edcreg.web.intercepter;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;
import com.alibaba.fastjson.JSON;
import com.cmos.edcreg.beans.common.OutputObject;
import com.cmos.edcreg.utils.StringUtil;
import com.cmos.edcreg.utils.des.DesSpecial;
import com.cmos.edcreg.utils.enums.ReturnInfoEnums;
import com.fasterxml.jackson.databind.ObjectMapper;

/**
 * 对响应报文统一处理,对bean内容进行加密
 * @author qiaozhong
 */
@Component
//声明该类要处理的包路径
@ControllerAdvice("com.cmos.edcreg.web.controller")
public class ResponseAdvice implements ResponseBodyAdvice {
    
    private final Logger logger = LoggerFactory.getLogger(ResponseAdvice.class);
    
    /* 
     * 对response处理的具体方法
     * arg0为返回的报文体,arg0为org.json.jsonObject,使用alibaba.json方法转换时候报错了
     */
    @Override
    public Object beforeBodyWrite(Object arg0, MethodParameter arg1,
            MediaType arg2, Class arg3, ServerHttpRequest arg4,
            ServerHttpResponse arg5) {
        OutputObject out = new OutputObject();
        try {
            //arg0转换为OutputObject类型
            ObjectMapper objectMapper=new ObjectMapper();
            out = objectMapper.readValue(org.json.JSONObject.valueToString(arg0), OutputObject.class);
            //获取加密密钥
            String oldEncryptKey = out.getBean().get("oldEncryptKey");
            //获取加密字符串
            DesSpecial des = new DesSpecial();
            String encryptData = des.strEnc(JSON.toJSONString(out.getBean()), oldEncryptKey, null, null);
            //封装数据(清除原来数据,放入加密数据)
            out.getBean().clear();
            out.getBean().put("data", encryptData);
            return out;
        } catch (Exception e) {
            logger.error("返回报文处理出错", e);
            out.setReturnCode(ReturnInfoEnums.PROCESS_ERROR.getCode());
            out.setReturnMessage(ReturnInfoEnums.PROCESS_ERROR.getMessage());
            return out;
        }
    }

    /* 
     * 选择哪些类,或哪些方法需要走beforeBodyWrite
     * 从arg0中可以获取方法名和类名
     * arg0.getMethod().getDeclaringClass().getName()为获取方法名
     */
    @Override
    public boolean supports(MethodParameter arg0, Class arg1) {
        if("com.cmos.edcreg.web.controller.GdH5AppointmentActiveVideoNewController".equals(arg0.getMethod().getDeclaringClass().getName())){
            return true;
        }else{
            return false;
        }
    }
}
<think>嗯,用户问的是如何在Spring Boot中打印请求参数返回结果。首先,我得回想一下自己以前处理类似需求的经验。记得Spring Boot拦截器或者过滤器可以用,或者用AOP面向切面编程。不过具体怎么操作呢? 首先,打印请求参数。可能需要获取HTTP请求中的参数,比如GET请求的查询参数,或者POST请求的body体内容。对于GET请求,可以通过HttpServletRequest的getParameterMap来获取参数。对于POST请求,如果是表单提交,也可以用同样的方法,但如果是JSON格式,可能需要读取输入流来获取body内容。不过要注意,输入流一旦读取了,后续处理可能会出问题,所以可能需要用ContentCachingRequestWrapper来缓存请求内容。 然后是关于返回结果。返回结果可能在处理完控制器方法之后,所以可以在拦截器的afterCompletion方法里处理,或者使用ResponseBodyAdvice接口来拦截响应体。或者用AOP的环绕通知,在方法执行前后记录请求响应。 另外,用户可能需要简单的配置,所以是否有一些现成的库或者注解可以用?比如Spring自带的日志功能,或者使用actuator端点?不过可能用户想要的是自定义的日志输出,所以还是自己实现比较靠谱。 接下来,步骤大概是这样的:创建一个拦截器,在preHandle方法里记录请求的信息,包括URL、方法类型、参数等。对于POST的body,需要用缓存请求包装类,否则输入流只能读一次。然后在afterCompletion方法里记录响应的数据,这时候响应已经发送给客户端了,可以通过ContentCachingResponseWrapper来获取响应内容。 另外,ResponseBodyAdvice可以用来处理返回的对象,这样可以在写入响应之前记录日志。或者使用@ControllerAdvice配合@ResponseBody来修改响应,不过记录日志的话可能用这个不太合适。 对了,还有过滤器(Filter)的方式,比如使用OncePerRequestFilter,在过滤器中包装请求响应,然后记录日志。不过过滤器是在拦截器之前执行的,可能更底层一些。 具体实现的话,可能需要考虑以下几点: 1. 创建拦截器类实现HandlerInterceptor,重写preHandleafterCompletion方法。 2. 在preHandle中记录请求参数,处理GETPOST的不同情况。 3. 使用ContentCachingRequestWrapperContentCachingResponseWrapper来缓存请求响应的内容,以便多次读取。 4. 在afterCompletion中获取响应状态码响应内容。 5. 配置拦截器,将其注册到Spring Boot拦截器链中。 另外,处理请求体的时候需要注意字符编码问题,否则读取的内容可能会乱码。同时,文件上传之类的请求可能需要特殊处理,避免将文件内容全部记录到日志中,影响性能。 还有,响应内容如果是JSON格式,可能需要将其转换为字符串记录。使用Jackson的ObjectMapper可以做到这一点,或者直接读取缓存的响应内容字节数组,转换成字符串。 测试的时候,可能需要发送不同的请求类型(GET、POST)来验证日志是否正确输出。同时,确保拦截器不会对性能造成太大影响,特别是在高并发的情况下,日志记录应该尽量高效。 可能用户还希望排除某些特定的路径,比如健康检查的端点,或者静态资源,这时候在注册拦截器时可以配置排除的路径模式。 总结下来,实现方法大致有两种:拦截器配合包装类,或者使用AOP。拦截器的方法可能更直接,而AOP可以更灵活地针对特定的控制器方法进行处理。不过用户可能更倾向于使用拦截器的方式,因为能统一处理所有请求。 最后,代码示例应该包括拦截器的实现、包装类的使用、以及如何注册拦截器。同时提醒用户注意性能大请求体的处理,避免内存溢出。</think>在 Spring Boot 中打印请求参数返回结果,可以通过以下两种常用方式实现: --- ### **方法一:使用拦截器(Interceptor)** 通过实现 `HandlerInterceptor` 接口,结合请求/响应包装类,记录请求响应信息。 #### 1. 创建拦截器类 ```java import org.springframework.stereotype.Component; import org.springframework.web.servlet.HandlerInterceptor; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; @Component public class LogInterceptor implements HandlerInterceptor { // 请求处理前记录参数 @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) { // 包装请求以缓存请求体(用于后续读取) if (request instanceof ContentCachingRequestWrapper) { ContentCachingRequestWrapper requestWrapper = (ContentCachingRequestWrapper) request; logRequest(requestWrapper); } return true; } // 请求完成后记录响应 @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { ContentCachingResponseWrapper responseWrapper = (ContentCachingResponseWrapper) response; logResponse(responseWrapper); } private void logRequest(ContentCachingRequestWrapper request) { String requestBody = new String(request.getContentAsByteArray(), StandardCharsets.UTF_8); System.out.println("请求参数: " + requestBody); } private void logResponse(ContentCachingResponseWrapper response) { String responseBody = new String(response.getContentAsByteArray(), StandardCharsets.UTF_8); System.out.println("返回结果: " + responseBody); } } ``` #### 2. 配置拦截器注册 ```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 WebConfig implements WebMvcConfigurer { @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(new LogInterceptor()) .addPathPatterns("/**") .excludePathPatterns("/static/**"); // 排除静态资源 } } ``` #### 3. 配置请求/响应包装过滤器 确保请求响应被包装,以便多次读取内容: ```java import javax.servlet.*; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.stereotype.Component; import org.springframework.web.util.ContentCachingRequestWrapper; import org.springframework.web.util.ContentCachingResponseWrapper; @Component public class ContentCachingFilter implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) { HttpServletRequest req = new ContentCachingRequestWrapper((HttpServletRequest) request); HttpServletResponse res = new ContentCachingResponseWrapper((HttpServletResponse) response); chain.doFilter(req, res); // 确保响应内容写入客户端 res.copyBodyToResponse(); } } ``` --- ### **方法二:使用 AOP 面向切面** 通过 AOP 拦截 Controller 方法,记录入参返回值。 #### 1. 添加 AOP 依赖 ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency> ``` #### 2. 定义切面类 ```java import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.springframework.stereotype.Component; @Aspect @Component public class LogAspect { @Around("execution(* com.example.demo.controller.*.*(..))") // 拦截指定包下的Controller public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable { // 打印请求参数 Object[] args = joinPoint.getArgs(); System.out.println("请求参数: " + Arrays.toString(args)); // 执行方法并获取返回值 Object result = joinPoint.proceed(); // 打印返回结果 System.out.println("返回结果: " + result); return result; } } ``` --- ### **注意事项** 1. **性能影响**:频繁记录大请求体或响应体可能影响性能,建议生产环境按需使用。 2. **敏感信息**:避免打印敏感数据(如密码、Token)。 3. **文件上传**:不适用于文件上传请求(避免内存溢出)。 --- 两种方法各有优劣: - **拦截器**:适合全局统一处理,但需处理请求/响应包装。 - **AOP**:更灵活,可针对特定 Controller 或方法,但需依赖 AOP 配置。 根据实际场景选择合适方案即可。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值