5

我使用 slf4j 登录我的项目。我想使用 MDC 来记录用户 ID 参数。所以我检查了教程和文档,并制作了如下代码:

MDC.put("key", userId)

userId 实际上是一个字符串。

我使用我常用的 log4j xml 和 appenders 等属性。添加 %X{key} 但没有任何反应。我根本看不到任何代替 %X{key} 的东西,但是 %-5p 或 %c 等其他参数效果很好。

所以我用调试来观察 MDC.put() 方法中发生了什么,并发现在 MDC 的初始化中使用了:

MDCAdapter mdcAdapter;
mdcAdapter = StaticMDCBinder.SINGLETON.getMDCA();

IDEA 中的调试表明它具有“Log4jMDCAdapter”,就像 MDCAdapter 的实现之一。但后来我看看StaticMDCBinder,有这样的代码:

public MDCAdapter getMDCA() {
   return new NOPMDCAdapter();
}

那么 slf4j 怎么可能用适当的适配器初始化 MDC,例如 log4j ???我没明白。因为它总是使用 NOPMDCAdapter 它不能在 MDC 中存储任何东西,也不能在日志中显示它。我该如何解决?

在类路径中我有:

  • log4j-1.2.16.jar
  • slf4j-api-1.6.1.jar
  • slf4j-api-1.6.2.jar
  • slf4j-jcl-1.6.2.jar
  • slf4j-log4j12-1.6.2.jar
4

2 回答 2

1

SLF4J 的 MDC 只是一个门面,用于切换基于底层 logger 的 MDC 的适用实现。如果它没有从记录器中找到任何合适的 MDCAdapter,它将回退到 NO-OP 适配器。为此,它尝试查找名为 StaticMDCBinder https://github.com/qos-ch/slf4j/blob/v_1.7.25/slf4j-api/src/main/java/org/slf4j/MDC.java#L99的类

因此,在这种情况下,如@Wizzard 的回答中所述,如果您的类路径中有两个 StaticMDCBinder 类,tomcat 将选择其中一个。

slf4j-simple 确实有一个 StaticMDCBinder,它默认为 NO-OP 适配器,因此删除 slf4j-simple jar 应该可以解决这里的问题。

于 2021-06-18T13:16:09.053 回答
0

问题解决了。它是项目中使用的其他库的超越依赖。即slf4j-jcl,我将部分添加到maven pom中,现在它工作正常。

于 2013-01-17T07:57:35.593 回答