slf4j:如何记录格式化消息,对象数组,异常

row*_*owe 261 java logging exception slf4j

记录填充消息和异常堆栈跟踪的正确方法是什么?

logger.error(
    "\ncontext info one two three: {} {} {}\n",
    new Object[] {"1", "2", "3"},
    new Exception("something went wrong"));
Run Code Online (Sandbox Code Playgroud)

我想生成类似于此的输出:

context info one two three: 1 2 3
java.lang.Exception: something went wrong
stacktrace 0
stacktrace 1
stacktrace ...
Run Code Online (Sandbox Code Playgroud)

slf4j版本1.6.1

Cek*_*eki 404

从SLF4J 1.6.0开始,如果存在多个参数,并且如果日志语句中的最后一个参数是异常,则SLF4J将假定用户希望将最后一个参数视为异常而不是简单参数.另请参阅相关的FAQ条目.

所以,写(在SLF4J版本1.7.x及更高版本中)

 logger.error("one two three: {} {} {}", "a", "b", 
              "c", new Exception("something went wrong"));
Run Code Online (Sandbox Code Playgroud)

或写作(在SLF4J版本1.6.x中)

 logger.error("one two three: {} {} {}", new Object[] {"a", "b", 
              "c", new Exception("something went wrong")});
Run Code Online (Sandbox Code Playgroud)

会屈服

one two three: a b c
java.lang.Exception: something went wrong
    at Example.main(Example.java:13)
    at java.lang.reflect.Method.invoke(Method.java:597)
    at ...
Run Code Online (Sandbox Code Playgroud)

确切的输出将取决于底层框架(例如logback,log4j等)以及底层框架的配置方式.但是,如果最后一个参数是异常,则无论底层框架如何,它都将被解释为此类参数.

  • 我要给@Ceki一个很难的时间,因为它不是在Javadocs中,而是它在`Logger` javadoc类的顶部:http://www.slf4j.org/apidocs/org/slf4j/Logger.html (6认同)
  • 您使用的是哪种基础日志框架?正如我在上面的回答中所提到的,如果最后一个参数是一个例外,它将被解释为无论底层框架如何.(经过logback测试,slf4j-log4j12,slf4j-jdk14和slf4j-simple.) (4认同)
  • 抱歉,我没有意识到在您的示例中,您在格式字符串中使用了n = 3个占位符,并且在对象数组中使用了n + 1 = 4个元素.我在格式字符串中有n个占位符,在对象数组中也有n个元素,另外还有一个异常作为第三个参数.我的期望是异常将使用stacktrace打印,但这种情况从未发生过.这是否按设计工作?另外,如果我在对象数组中有n个占位符和n个元素,异常是最后一个元素,我看不到任何堆栈跟踪.也许应该更加强调数组中n + 1个对象的n个占位符. (3认同)
  • 我创建了[改进请求](https://jira.qos.ch/browse/SLF4J-482),如果你喜欢,可以投票。 (2认同)

Yan*_*niv 7

除了@Ceki的回答,如果您正在使用logback并在项目中设置配置文件(通常是logback.xml),您可以定义日志以绘制堆栈跟踪以及使用

<encoder>
    <pattern>%date |%-5level| [%thread] [%file:%line] - %msg%n%ex{full}</pattern> 
</encoder>
Run Code Online (Sandbox Code Playgroud)

模式中的%ex是产生差异的原因