当前位置: 首页 > news >正文

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自定义异常处理
  • 关于类与构造函数继承的小挑战
  • bhyve安装win10第一篇:如何下载Win10 iso安装光盘
  • Java, 将 csv 中空值用上一行的值填充
  • c++异常处理(c++11版)与智能指针 SmartPtr 的应用(主讲shared_ptr浅实现)
  • “双指针”算法下篇
  • zabbix监控进程、日志、主从(状态、延迟)
  • spring security怎么解决用户的权限问题
  • 速盾:海外cdn加速可以https加密吗?
  • 数据库集群技术
  • Flask-RESTFul 之 RESTFul 的响应处理 之定制返回的 json格式
  • vue3插件原理
  • 监控领域的物理对抗攻击综述——Physical Adversarial Attacks for Surveillance: A Survey
  • 【附node操作实例】redis简明入门系列—字符串类型
  • 【跃迁之路】【735天】程序员高效学习方法论探索系列(实验阶段492-2019.2.25)...
  • angular学习第一篇-----环境搭建
  • Apache Spark Streaming 使用实例
  • Asm.js的简单介绍
  • extract-text-webpack-plugin用法
  • Git 使用集
  • Git同步原始仓库到Fork仓库中
  • go append函数以及写入
  • javascript从右向左截取指定位数字符的3种方法
  • Javascript设计模式学习之Observer(观察者)模式
  • Just for fun——迅速写完快速排序
  • NSTimer学习笔记
  • Python 使用 Tornado 框架实现 WebHook 自动部署 Git 项目
  • REST架构的思考
  • spring cloud gateway 源码解析(4)跨域问题处理
  • 测试开发系类之接口自动化测试
  • 初探 Vue 生命周期和钩子函数
  • 从零开始学习部署
  • 解决jsp引用其他项目时出现的 cannot be resolved to a type错误
  • 京东美团研发面经
  • 开发基于以太坊智能合约的DApp
  • 开发了一款写作软件(OSX,Windows),附带Electron开发指南
  • 跨域
  • 微信支付JSAPI,实测!终极方案
  • 我有几个粽子,和一个故事
  • 用Visual Studio开发以太坊智能合约
  • PostgreSQL之连接数修改
  • 湖北分布式智能数据采集方法有哪些?
  • ​埃文科技受邀出席2024 “数据要素×”生态大会​
  • ​你们这样子,耽误我的工作进度怎么办?
  • # Redis 入门到精通(九)-- 主从复制(1)
  • (2024,RWKV-5/6,RNN,矩阵值注意力状态,数据依赖线性插值,LoRA,多语言分词器)Eagle 和 Finch
  • (CVPRW,2024)可学习的提示:遥感领域小样本语义分割
  • (delphi11最新学习资料) Object Pascal 学习笔记---第13章第1节 (全局数据、栈和堆)
  • (附源码)计算机毕业设计SSM疫情居家隔离服务系统
  • (附源码)小程序 交通违法举报系统 毕业设计 242045
  • (中等) HDU 4370 0 or 1,建模+Dijkstra。
  • **PHP二维数组遍历时同时赋值
  • .dwp和.webpart的区别