一文吃透!Spring Boot 项目请求日志记录,这几招你绝对不能错过!
wptr33 2025-06-30 20:45 54 浏览
在互联网应用开发的高速赛道上,系统的稳定性、可维护性以及安全性是每一位开发者都必须关注的核心要素。而请求日志记录,就如同系统的 “黑匣子”,能够为我们提供排查故障、分析用户行为、优化系统性能等关键信息,在整个开发流程中占据着举足轻重的地位。对于广大 Spring Boot 开发者而言,如何快速、高效地实现请求日志记录,一直是日常开发工作中亟待解决的问题。
在过往的开发实践中,我们已经积累了一些常见的请求日志记录方式,它们各有特点,在不同的场景下发挥着重要作用。
一、传统的请求日志记录方式
1、使用拦截器
拦截器就像是一位忠诚的 “门卫”,在请求进入控制器之前和请求完成之后,都能执行特定的操作。
大致流程
- 创建一个自定义的拦截器类,让它实现 Spring 框架提供的 HandlerInterceptor 接口,从而具备拦截请求的能力。
- 重写 preHandle 方法,这个方法会在请求到达控制器之前被调用,我们可以在这个方法中记录请求的关键信息,比如请求的 URL、参数等,为后续的分析和排查提供依据。
- 重写 afterCompletion 方法,在请求处理完成之后,这个方法会被触发,我们可以在这里记录请求的处理结果、耗时等信息,全面了解请求的生命周期。
- 最后,将我们创建的拦截器注册到 Spring MVC 的配置中,这样拦截器就能在整个项目中发挥作用了。
2、使用过滤器
过滤器类似于一个 “关卡”,对所有经过的请求和响应进行统一的处理。
大致流程
- 创建一个自定义的过滤器类,实现 Filter 接口,使其能够对请求和响应进行过滤操作。
- 在 doFilter 方法中,编写记录请求和响应日志的逻辑,这个方法会在请求和响应的处理过程中被调用,我们可以在这里获取并记录请求的详细信息,包括请求头、请求体等。
- 将过滤器注册到 Servlet 容器中,确保它能够对所有的请求和响应生效。
3、使用AOP
AOP(面向切面编程)就像是一把 “手术刀”,能够精准地切入到方法的执行过程中,实现对日志记录的统一管理。
大致流程
- 定义一个切面类,使用 @Aspect 注解将其标记为切面,表明这个类将用于处理横切关注点。
- 使用 @Around 注解定义切点,切点就像是一个 “钩子”,可以在方法执行前、执行后或抛出异常后,执行我们定义的日志记录逻辑,灵活地记录不同阶段的信息。
- 将切面类注册到 Spring 容器中,使其能够在项目中发挥作用。
这些传统的方式虽然有效,但在实际应用中可能会面临一些配置繁琐、代码冗余等问题。那么,有没有更简单、高效的方法呢?答案是肯定的!Spring Boot 2 为我们提供了一些内置的超强请求记录方法,让我们一起来看看吧!
二、Spring Boot 2 内置的超便捷请求记录方法
方法一:通过 Spring Boot Actuator 的 httptrace 端点
步骤如下:
1、项目pom引入Actuator GAV
在项目的 pom 文件中添加以下依赖,引入 Spring Boot Actuator:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
2、开启httptrace端点
在配置文件中,我们可以通过以下配置开启所有端点,以便使用 httptrace 端点:
management:
endpoints:
web:
exposure:
include: '*'
3、测试
编写一个测试 controller,用于模拟请求:
@RequestMapping("echo")
@RestController
public class EchoController {
@RequestMapping("say/{sleep}")
public String echo(@PathVariable("sleep") long sleep, String msg) {
if(sleep == 0){
throw new IllegalArgumentException("sleep参数不能为0");
}
try {
TimeUnit.MILLISECONDS.sleep(sleep);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
return "echo:" + msg;
}
}
浏览器先访问测试控制器
http://localhost:8080/echo/say/100?msg=hello
再访问下httptrace端点
http://localhost:8080/actuator/httptrace
这时,你会惊喜地发现,请求记录已经清晰地呈现在你面前。不过,需要注意的是,httptrace 信息默认存储在内存中,如果需要持久化存储,可以通过实现
org.springframework.boot.actuate.trace.http.HttpTraceRepository
,将其存储到其他存储介质上,如数据库、文件系统等。
方法二:通过监听 ServletRequestHandledEvent 事件
Spring 框架提供了丰富的事件机制,我们可以通过监听
ServletRequestHandledEvent 事件来记录请求日志。
示例代码如下:
@Slf4j
public class RequestLogEventListener {
@EventListener
public void listener(ServletRequestHandledEvent event){
log.info("request client ip :{},request method:{}",event.getClientAddress(),event.getMethod());
log.info("request url:{},cost time:{} ms",event.getRequestUrl(),event.getProcessingTimeMillis());
if(event.wasFailure()){
log.error("request fail,error msg:{}",event.getFailureCause());
}
}
}
访问测试控制器,然后观察控制台
你会发现请求日志已经被准确地记录下来。
ServletRequestHandledEvent 事件由
org.springframework.web.servlet.FrameworkServlet#processRequest
方法触发发送的,这为我们提供了一个非常方便的记录请求日志的时机。
开启 DispatcherServlet 日志,日志级别为debug
通过开启 DispatcherServlet 的日志,我们可以获取到详细的请求信息。
在配置文件中进行如下配置:
logging:
level:
org.springframework.web.servlet.DispatcherServlet: DEBUG
可以通过
http://localhost:8080/actuator/loggers/org.springframework.web.servlet.DispatcherServlet 动态调整日志级别,不过需要注意的是,这种方式在项目重启后,日志级别会恢复默认。示例命令如下:
curl -X POST http://localhost:8080/actuator/loggers/org.springframework.web.servlet.DispatcherServlet \
-H "Content-Type: application/json" \
-d '{"configuredLevel": "DEBUG"}'
最后访问测试控制器,观察控制台
你会发现有请求信息输出。但细心的你可能会发现 parameters={masked},这是因为 DispatcherServlet 默认会将请求参数进行掩码处理,以保护敏感信息不被泄露。不过,我们可以通过以下配置让请求参数显示出来:
spring:
http:
log-request-details: true
将请求参数也显示出来,再次访问测试控制器,观察控制台
会发现请求参数已经打印出来
三、总结
本文详细介绍了 Spring Boot 内置的三种记录请求日志的方式,这些方法不仅简单易用,而且功能强大,大大提高了我们的开发效率。如果想要将日志记录到存储介质,个人比较推荐使用
ServletRequestHandledEvent 这种方式,它的配置相对简单,并且能够满足大多数场景的需求。当然,如果有定制化的需求,比如需要记录响应体,那么开头介绍的那三种传统方式也是不错的选择,它们能够提供更灵活的扩展能力。但无论如何,利用好 Spring Boot 提供的这些内置功能,能够让我们的开发工作事半功倍。
四、demo链接
为了方便大家实践,文中所有代码的 demo 链接为:
https://github.com/lyb-geek/springboot-learning/tree/master/springboot-request-log。
赶紧动手实践一下吧,相信你一定会对 Spring Boot 的请求日志记录有更深入的理解!
希望这篇文章能够帮助你在 Spring Boot 开发中更好地记录请求日志,提升系统的可维护性和稳定性。如果在实践过程中有任何问题,欢迎随时交流探讨!
相关推荐
- MySQL进阶五之自动读写分离mysql-proxy
-
自动读写分离目前,大量现网用户的业务场景中存在读多写少、业务负载无法预测等情况,在有大量读请求的应用场景下,单个实例可能无法承受读取压力,甚至会对业务产生影响。为了实现读取能力的弹性扩展,分担数据库压...
- 3分钟短文 | Laravel SQL筛选两个日期之间的记录,怎么写?
-
引言今天说一个细分的需求,在模型中,或者使用laravel提供的EloquentORM功能,构造查询语句时,返回位于两个指定的日期之间的条目。应该怎么写?本文通过几个例子,为大家梳理一下。学习时...
- 一文由浅入深带你完全掌握MySQL的锁机制原理与应用
-
本文将跟大家聊聊InnoDB的锁。本文比较长,包括一条SQL是如何加锁的,一些加锁规则、如何分析和解决死锁问题等内容,建议耐心读完,肯定对大家有帮助的。为什么需要加锁呢?...
- 验证Mysql中联合索引的最左匹配原则
-
后端面试中一定是必问mysql的,在以往的面试中好几个面试官都反馈我Mysql基础不行,今天来着重复习一下自己的弱点知识。在Mysql调优中索引优化又是非常重要的方法,不管公司的大小只要后端项目中用到...
- MySQL索引解析(联合索引/最左前缀/覆盖索引/索引下推)
-
目录1.索引基础...
- 你会看 MySQL 的执行计划(EXPLAIN)吗?
-
SQL执行太慢怎么办?我们通常会使用EXPLAIN命令来查看SQL的执行计划,然后根据执行计划找出问题所在并进行优化。用法简介...
- MySQL 从入门到精通(四)之索引结构
-
索引概述索引(index),是帮助MySQL高效获取数据的数据结构(有序),在数据之外,数据库系统还维护者满足特定查询算法的数据结构,这些数据结构以某种方式引用(指向)数据,这样就可以在这些数据结构...
- mysql总结——面试中最常问到的知识点
-
mysql作为开源数据库中的榜一大哥,一直是面试官们考察的重中之重。今天,我们来总结一下mysql的知识点,供大家复习参照,看完这些知识点,再加上一些边角细节,基本上能够应付大多mysql相关面试了(...
- mysql总结——面试中最常问到的知识点(2)
-
首先我们回顾一下上篇内容,主要复习了索引,事务,锁,以及SQL优化的工具。本篇文章接着写后面的内容。性能优化索引优化,SQL中索引的相关优化主要有以下几个方面:最好是全匹配。如果是联合索引的话,遵循最...
- MySQL基础全知全解!超详细无废话!轻松上手~
-
本期内容提醒:全篇2300+字,篇幅较长,可搭配饭菜一同“食”用,全篇无废话(除了这句),干货满满,可收藏供后期反复观看。注:MySQL中语法不区分大小写,本篇中...
- 深入剖析 MySQL 中的锁机制原理_mysql 锁详解
-
在互联网软件开发领域,MySQL作为一款广泛应用的关系型数据库管理系统,其锁机制在保障数据一致性和实现并发控制方面扮演着举足轻重的角色。对于互联网软件开发人员而言,深入理解MySQL的锁机制原理...
- Java 与 MySQL 性能优化:MySQL分区表设计与性能优化全解析
-
引言在数据库管理领域,随着数据量的不断增长,如何高效地管理和操作数据成为了一个关键问题。MySQL分区表作为一种有效的数据管理技术,能够将大型表划分为多个更小、更易管理的分区,从而提升数据库的性能和可...
- MySQL基础篇:DQL数据查询操作_mysql 查
-
一、基础查询DQL基础查询语法SELECT字段列表FROM表名列表WHERE条件列表GROUPBY分组字段列表HAVING分组后条件列表ORDERBY排序字段列表LIMIT...
- MySql:索引的基本使用_mysql索引的使用和原理
-
一、索引基础概念1.什么是索引?索引是数据库表的特殊数据结构(通常是B+树),用于...
- 一周热门
-
-
C# 13 和 .NET 9 全知道 :13 使用 ASP.NET Core 构建网站 (1)
-
程序员的开源月刊《HelloGitHub》第 71 期
-
详细介绍一下Redis的Watch机制,可以利用Watch机制来做什么?
-
假如有100W个用户抢一张票,除了负载均衡办法,怎么支持高并发?
-
Java面试必考问题:什么是乐观锁与悲观锁
-
如何将AI助手接入微信(打开ai手机助手)
-
redission YYDS spring boot redission 使用
-
SparkSQL——DataFrame的创建与使用
-
一文带你了解Redis与Memcached? redis与memcached的区别
-
如何利用Redis进行事务处理呢? 如何利用redis进行事务处理呢英文
-
- 最近发表
- 标签列表
-
- git pull (33)
- git fetch (35)
- mysql insert (35)
- mysql distinct (37)
- concat_ws (36)
- java continue (36)
- jenkins官网 (37)
- mysql 子查询 (37)
- python元组 (33)
- mybatis 分页 (35)
- vba split (37)
- redis watch (34)
- python list sort (37)
- nvarchar2 (34)
- mysql not null (36)
- hmset (35)
- python telnet (35)
- python readlines() 方法 (36)
- munmap (35)
- docker network create (35)
- redis 集合 (37)
- python sftp (37)
- setpriority (34)
- c语言 switch (34)
- git commit (34)