JAVA之MAC详解以及子线程MDC传递

2024-08-24 09:04
文章标签 java 线程 详解 mac 传递 mdc

本文主要是介绍JAVA之MAC详解以及子线程MDC传递,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

MDC简介

MDC(Mapped Diagnostic Context)是用于分布式系统中跟踪和诊断日志的重要概念。是一个在Java项目中用于日志跟踪的工具,它允许你在多线程环境下关联和传递特定的上下文信息。
MDC是一个线程本地的、可维护的、可传递的上下文环境。在Java中,MDC主要用于在应用程序的不同组件之间传递日志上下文信息,例如用户会话ID,请求ID,用户身份信息等。MDC让你可以将这些信息关联到特定的日志事件中,以便后续的日志处理器(如日志输出器)能够在日志中显示或处理这些信息。

MDC原理

MDC的实现原理通常基于线程本地变量(ThreadLocal),每个线程都有自己的MDC,线程在处理请求时可以将上下文信息设置到MDC中,这些信息会和该线程相关联。当日志事件发生时,日志框架会从MDC中获取相应的上下文信息,并将其包含在日志中。

作用

MDC的主要作用是在日志输出时,自动将当前线程的上下文信息,也就是设置在MDC中的信息,添加到日志中。这样可以帮助我们更好地理解系统的行为,特别是在跨线程、跨进程,甚至跨服务器的情况下,轻松地追踪某个请求的完整生命周期。

  • 跟踪日志上下文信息:MDC允许在日志中添加额外的上下文信息,帮助理解日志事件发生的背景和环境;
  • 诊断和调试:在多线程环境中,使用MDC可以将特定的上下文信息关联到日志中,有助于排查问题和调试程序。
  • 日志过滤和路由:MDC中的上下文信息可以被日志处理器用来过滤,路由或分类日志事件,例如基金用户会话ID将日志事件路由到特定的日志文件或系统。

基本使用过程

使用MDC的步骤通常如下:

  1. 设置MDC中的信息
    通过普通代码、拦截器或者AOP在方法调用链最开始,设置MDC的值。例如:
MDC.put("traceId","12345678");
MDC.put("requestId","request-998");
  1. 定义日志格式,其中%X{}代表去MDC取值,例如:
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} %thread | %X{traceId} | %X{requestId} | %-5level %logger{50} %msg%n</pattern>
  1. 清除MDC中的信息
    当不再需要MDC中的信息时,可以通过调用MDC的remove方法来清除它们。例如:
MDC.remove("traceId");
MDC.remove("requestId");

子线程MDC传递

既然我们知道MDC底层使用TreadLocal来实现,那根据TreadLocal的特点,它是可以让我们在同一个线程中共享数据的,但是往往我们在业务方法中,会开启多线程来执行程序,这样的话MDC就无法传递到其他子线程了。这时,我们需要使用额外的方法来传递存在TreadLocal里的值。MDC提供了一个叫getCopyOfContextMap的方法,很显然,该方法就是把当前线程TreadLocal绑定的Map获取出来,之后就是把该Map绑定到子线程中的ThreadLocal中了,具体代码如下:

Map<String, String> copyOfContextMap = MDC.getCopyOfContextMap();
new Thread(() -> {if (copyOfContextMap != null) {MDC.setContextMap(copyOfContextMap);}log.info("这个是子线程的信息");
}).start();

也就是说,我们在主线程中获取MDC的值,然后在子线程中设置进去,这样,子线程打印的信息也会带有整个调用链共同的traceId了。

这篇关于JAVA之MAC详解以及子线程MDC传递的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1102039

相关文章

MySQL中的分组和多表连接详解

《MySQL中的分组和多表连接详解》:本文主要介绍MySQL中的分组和多表连接的相关操作,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录mysql中的分组和多表连接一、MySQL的分组(group javascriptby )二、多表连接(表连接会产生大量的数据垃圾)MySQL中的

Java 实用工具类Spring 的 AnnotationUtils详解

《Java实用工具类Spring的AnnotationUtils详解》Spring框架提供了一个强大的注解工具类org.springframework.core.annotation.Annot... 目录前言一、AnnotationUtils 的常用方法二、常见应用场景三、与 JDK 原生注解 API 的

Java controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

Java中的StringBuilder之如何高效构建字符串

《Java中的StringBuilder之如何高效构建字符串》本文将深入浅出地介绍StringBuilder的使用方法、性能优势以及相关字符串处理技术,结合代码示例帮助读者更好地理解和应用,希望对大家... 目录关键点什么是 StringBuilder?为什么需要 StringBuilder?如何使用 St

使用Java将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

redis中使用lua脚本的原理与基本使用详解

《redis中使用lua脚本的原理与基本使用详解》在Redis中使用Lua脚本可以实现原子性操作、减少网络开销以及提高执行效率,下面小编就来和大家详细介绍一下在redis中使用lua脚本的原理... 目录Redis 执行 Lua 脚本的原理基本使用方法使用EVAL命令执行 Lua 脚本使用EVALSHA命令

Java并发编程之如何优雅关闭钩子Shutdown Hook

《Java并发编程之如何优雅关闭钩子ShutdownHook》这篇文章主要为大家详细介绍了Java如何实现优雅关闭钩子ShutdownHook,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 目录关闭钩子简介关闭钩子应用场景数据库连接实战演示使用关闭钩子的注意事项开源框架中的关闭钩子机制1.

Maven中引入 springboot 相关依赖的方式(最新推荐)

《Maven中引入springboot相关依赖的方式(最新推荐)》:本文主要介绍Maven中引入springboot相关依赖的方式(最新推荐),本文给大家介绍的非常详细,对大家的学习或工作具有... 目录Maven中引入 springboot 相关依赖的方式1. 不使用版本管理(不推荐)2、使用版本管理(推

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows

在 Spring Boot 中实现异常处理最佳实践

《在SpringBoot中实现异常处理最佳实践》本文介绍如何在SpringBoot中实现异常处理,涵盖核心概念、实现方法、与先前查询的集成、性能分析、常见问题和最佳实践,感兴趣的朋友一起看看吧... 目录一、Spring Boot 异常处理的背景与核心概念1.1 为什么需要异常处理?1.2 Spring B