问题描述
我想像这样将任意对象传递给threadcontext.put()
方法:
MyObject originalMessage = new MyObject(...);
threadcontext.put("originalMessage",originalMessage);
但是该方法仅允许String
作为其第二个参数:docs here。
有没有办法在Log4j的MDC中将任意对象放入log4j?或对此的替代?我想要的是允许进一步执行log.error()
来自动记录originalMessage
,而无需将其作为参数传递到任何地方。
此外,我无法自己调用toString()
,因为我会急切地评估该方法(在该对象中非常繁重),所以我希望将其调用延迟到需要时才进行,就像您使用时一样将任意对象传递给log.debug("{}",someObject)
。
解决方法
代替它(不会编译):
ThreadContext.put("originalMessage",originalMessage);
我可以做到:
final var threadContextMap = (ObjectThreadContextMap) ThreadContext.getThreadContextMap();
threadContextMap.putValue("originalMessage",originalMessage);
只要您添加此JVM选项,此方法就起作用:
-Dlog4j2.threadContextMap=org.apache.logging.log4j.spi.CopyOnWriteSortedArrayThreadContextMap
现在,这看起来很骇人,因为getThreadContextMap()该方法应该返回内部地图的只读视图:
返回内部数据结构的只读视图,该内部数据结构用于存储线程上下文键值对...
因此,此解决方案正在投射只读地图,并且知道实现不是只读的,因此可以对其进行修改……不是最透明的解决方案,但它可以工作。