Spring boot 拦截所有请求的出参入参 并打印日志

不喜欢讲废话直接正题,
在开发阶段我们可能需要对所有的出参和入参打印log
在每个方法上都自己手动操作过于麻烦,下面介绍一个省事的办法。

自己封装request 和response

spring boot 提供了两个包装类型分别是:HttpServletRequestWrapper和HttpServletResponseWrapper

HttpServletRequestWrapper实例:

public class MyRequestWrapper  extends HttpServletRequestWrapper {

    private byte[] requestBody=null;

    @SneakyThrows
    public MyRequestWrapper(HttpServletRequest request) {
        super(request);
        String body = HttpUtils.getBody(request);

        String url = request.getRequestURI();
        log.info("访问路径{} 方法入参: {}",url,body);

        requestBody=body.getBytes(Charset.defaultCharset());

    }


    @Override
    public ServletInputStream getInputStream() throws IOException {
        final ByteArrayInputStream inputStream=new ByteArrayInputStream(requestBody);
        return new ServletInputStream() {
            @Override
            public boolean isFinished() {
                return false;
            }

            @Override
            public boolean isReady() {
                return false;
            }

            @Override
            public void setReadListener(ReadListener readListener) {

            }

            @Override
            public int read() throws IOException {
                return inputStream.read();
            }
        };
    }

    @Override
    public BufferedReader getReader() throws IOException {
        return new BufferedReader(new InputStreamReader(this.getInputStream()));
    }

    public byte[] getRequestBody(){
        return this.requestBody;
    }
}

大家都知道request body 只能取一次,所以这里的主要包装是用于将请求体数据自己保存起来可以多次使用!

HttpServletResponseWrapper示例:

public class ResponseWrapper extends HttpServletResponseWrapper {

    ByteArrayOutputStream output;

    public ResponseWrapper(HttpServletResponse response) {
        super(response);
        output = new ByteArrayOutputStream();
    }

    @Override
    public ServletOutputStream getOutputStream() throws IOException {
        return new ServletOutputStream() {
            @Override
            public boolean isReady() {
                return false;
            }

            @Override
            public void setWriteListener(WriteListener listener) {

            }

            @Override
            public void write(int b) throws IOException {
                output.write(b);
            }

            @Override
            public void write(byte[] b) throws IOException {
                output.write(b);
            }

            @Override
            public void write(byte[] b, int off, int len) throws IOException {
                output.write(b,off,len);
            }
        };
    }

    public byte[] getDataStream() {
        return output.toByteArray();
    }

} 

这里主要把response的数据用字节流读取保存,目的是一样的由于字节流的特性可以多次使用

封装完两个类之后开始配置Filter 主要目的是把原来的request和response换成我们封装的

示例:

@Component
@Slf4j
@WebFilter(filterName = "channelFilter", urlPatterns = {"/*"})
public class ChannelFilter  implements Filter {

    @Override
    /**
     * @author: Mir Dai
     * @description:  用自己封装request 和response入参
     * @date: 2021/5/20 15:39
     * @param request
     * @param response
     * @param chain
     * @return {@link Void}
     */
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain){
        try {
            ResponseWrapper responseWrapper = new ResponseWrapper((HttpServletResponse) response);
            ServletRequest requestWrapper = new MyRequestWrapper((HttpServletRequest) request);
            chain.doFilter(requestWrapper,responseWrapper);
            String s = new String(responseWrapper.getDataStream());
            String requestURI = ((HttpServletRequest) request).getRequestURI();
            log.info("访问路径{} 方法出参: {}",requestURI,s);
            response.getOutputStream().write(s.getBytes());
        }catch (Exception e){
            e.getMessage();
            log.error(e.getMessage());
        }
    }
}

这里有个细节,我们是用我们封装的response去读取我们业务返回的数据 然后用原来的response去返回数据
如果想更完善一点 可以把我们封装的response 所有请求头取出给原生的response头全部换掉再返回

这样我们的日志入参出参请求就完成了
在这里插入图片描述
在这里插入图片描述

有问题就是插入的姿势不对!

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Spring Boot中,可以使用拦截器来对请求进行拦截和处理。拦截器是一种AOP(面向切面编程)的技术,在请求到达控制器之前或之后对请求进行处理。拦截器可以用于实现一些通用的功能,如日志记录、权限验证、数校验等。 在Spring Boot中定义拦截器需要以下步骤: 1. 创建一个拦截器类,该类需要实现HandlerInterceptor接口。可以在拦截器中重写preHandle、postHandle和afterCompletion等方法来实现请求的前置处理、后置处理和完成处理。 2. 创建一个拦截器配置类,该类需要实现WebMvcConfigurer接口,并重写addInterceptors方法。在该方法中,可以注册拦截器类,并为其定义拦截规则。 在配置类中,可以通过使用addPathPatterns方法来定义需要拦截请求路径,也可以使用excludePathPatterns方法来排除某些请求路径不被拦截拦截器与过滤器在功能和使用上有一些区别: 1. 归属不同:过滤器属于Servlet技术,而拦截器属于SpringMVC技术。 2. 内容不同:过滤器对所有访问进行增强,而拦截器仅针对SpringMVC的访问进行增强。 通过上述步骤定义和配置拦截器,可以在Spring Boot应用中实现对请求拦截和处理。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [Spring-Boot 拦截器](https://blog.csdn.net/PeterMrWang/article/details/123916751)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [Springboot——拦截器](https://blog.csdn.net/weixin_51351637/article/details/128058053)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值