使用MDC实现日志链路跟踪
程序员文章站
2022-03-04 13:06:21
目录1.原理2.实现3.过滤器4.logback.xml5.返回体6.效果日志前言:在微服务环境中,我们经常使用skywalking、cat等去实现整体请求链路的追踪,但是这个整体运维成本高,架构复杂...
前言:
在微服务环境中,我们经常使用skywalking
、cat等去实现整体请求链路的追踪,但是这个整体运维成本高,架构复杂,我们来使用mdc
通过log来实现一个轻量级的会话事务跟踪功能。
1.原理
mdc org.sl4j.mdc
其实内部就是threadlocal,mdc提供了put/get/clear等几个核心接口,用于操作threadlocal中的数据;threadlocal中的k-v,可以在logback.xml中声明,最终将会打印在日志中。
// java代码 mdc.put("userid","laker"); // logback.xml %x{userid}
例如:
<property name="pattern" value="%d{hh:mm:ss.sss} [%thread] %-5level [%x{userid}] %logger{20} - %msg%n"/>
2.实现
整体流程如下:
- 用户登录系统,我们日志中记录
userid:laker
。 - 用户发起请求,一个请求中可能实际产生多个http请求,这里可以前端生成一个
requestid
- 在返回体中,返回
requestid
。 - 研发运维人员,可以根据
userid
和requestid
去日志中捞请求链路。
3.过滤器
@order(value = ordered.highest_precedence + 100) @component @webfilter(filtername = "mdcfilter", urlpatterns = "/*") public class mdcfilter extends onceperrequestfilter { @override protected void dofilterinternal(httpservletrequest httpservletrequest, httpservletresponse httpservletresponse, filterchain filterchain) throws servletexception, ioexception { try { mdc.put("userid", "laker"); mdc.put("requestid", idutil.fastuuid()); } catch (exception e) { // } try { filterchain.dofilter(httpservletrequest, httpservletresponse); } finally { mdc.clear(); } } }
4.logback.xml
<?xml version="1.0" encoding="utf-8"?> <configuration> <property name="log_home" value="logs"/> <property name="encoding" value="utf-8"/> <appender name="default" class="ch.qos.logback.core.rolling.rollingfileappender"> <file>${log_home}/test.log</file> <append>true</append> <prudent>false</prudent> <encoder class="ch.qos.logback.classic.encoder.patternlayoutencoder"> <pattern>%d{yyyy-mm-dd hh:mm:ss.sss} [%t] %-5level %logger{50} %line - %m%n</pattern> </encoder> <!-- 按天回滚 daily --> <rollingpolicy class="ch.qos.logback.core.rolling.timebasedrollingpolicy"> <!--归档日志文件名--> <filenamepattern>${log_home}/test.log.%d{yyyy-mm-dd}</filenamepattern> <!-- 最多保存15天历史文件 --> <maxhistory>15</maxhistory> </rollingpolicy> </appender> <!-- 日志输出格式 --> <property name="log.pattern" value="%d{hh:mm:ss.sss} [%thread] %-5level [%x{userid}|%x{requestid}] %logger{20} - [%method,%line] - %msg%n"/> <!-- 控制台输出 --> <appender name="console" class="ch.qos.logback.core.consoleappender"> <encoder> <pattern>${log.pattern}</pattern> </encoder> </appender> <appender name="stdout" class="ch.qos.logback.core.consoleappender"> <encoder class="ch.qos.logback.classic.encoder.patternlayoutencoder"> <pattern>%d{yyyy-mm-dd hh:mm:ss.sss} [%t] %-5level %logger{50} %line - %m%n</pattern> </encoder> </appender> <logger name="com.test.demo" level="debug"> <appender-ref ref="default"/> </logger> <!-- 日志输出级别 --> <root level="info"> <appender-ref ref="default"/> <appender-ref ref="console"/> </root> </configuration>
5.返回体
public class response<t> { @apimodelproperty(notes = "响应码,非200 即为异常", example = "200") private final int code; @apimodelproperty(notes = "响应消息", example = "提交成功") private final string msg; @apimodelproperty(notes = "响应数据") private final t data; @apimodelproperty(notes = "请求id") private final string requestid; public response(int code, string msg, t data) { this.code = code; this.msg = msg; this.data = data; this.requestid = mdc.get("requestid"); }
6.效果日志
响应:
{ code: 200, msg: "", requestid: "74a269a8-3cb4-417e-853c-b968b77cce23" }
日志:
18:37:15.997 [http-nio-8080-exec-1] info [laker|90717490-5ef4-4e46-bc2c-605952fc3803] c.l.m.c.infocontroller - [v2map,17] - null 18:37:38.980 [http-nio-8080-exec-2] info [laker|82bde351-f86e-466f-97a0-c857a0c4c1c9] c.l.m.c.infocontroller - [v2map,17] - null 18:37:39.992 [http-nio-8080-exec-3] info [laker|74a269a8-3cb4-417e-853c-b968b77cce23] c.l.m
到此这篇关于使用mdc实现日志链路跟踪的文章就介绍到这了,更多相关mdc实现日志链路跟踪内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!
推荐阅读
-
SpringBoot项目中通过MDC和自定义Filter操作traceId实现日志链路追踪
-
Spring Boot 中使用自定义注解,AOP 切面打印出入参日志及Dubbo链路追踪透传traceId
-
Dubbo全链路追踪日志的实现
-
spring cloud分布式使用日志链路跟踪
-
使用log4j MDC实现日志追踪
-
跟我学SpringCloud | 第十一篇:使用Spring Cloud Sleuth和Zipkin进行分布式链路跟踪
-
java开发中使用轻量级日志链路追踪框架 TLog
-
使用MDC实现日志链路跟踪
-
ASP.NET Core整合Zipkin链路跟踪的实现方法
-
spring cloud分布式使用日志链路跟踪