我不能在没有slf4j的情况下使用log4j吗?

时间:2015-03-27 12:36:40

标签: java maven logging log4j slf4j

我正在尝试将log4j用于作为JAR运行的简单小型maven应用程序。我理解slf4j有好处,但我试图尽可能简单地保持这一点,而不是诉诸System.out

我的pom.xml中有以下内容:

<dependency>
    <groupId>log4j</groupId>
    <artifactId>log4j</artifactId>
    <version>1.2.17</version>
</dependency>

它主要起作用,它正在我的log4j.properties指示记录到文件。但是......

...我在我的控制台中获得以下内容:

SLF4J: slf4j-api 1.6.x (or later) is incompatible with this binding.
SLF4J: Your binding is version 1.5.5 or earlier.
SLF4J: Upgrade your binding to version 1.6.x.
Exception in thread "main" java.lang.NoSuchMethodError: org.slf4j.impl.StaticLoggerBinder.getSingleton()Lorg/slf4j/impl/StaticLoggerBinder;
        at org.slf4j.LoggerFactory.bind(LoggerFactory.java:129)
        at org.slf4j.LoggerFactory.performInitialization(LoggerFactory.java:108)
        at org.slf4j.LoggerFactory.getILoggerFactory(LoggerFactory.java:302)
        at org.slf4j.LoggerFactory.getLogger(LoggerFactory.java:276)
        at org.apache.commons.logging.impl.SLF4JLogFactory.getInstance(SLF4JLogFactory.java:155)
        at org.apache.commons.logging.impl.SLF4JLogFactory.getInstance(SLF4JLogFactory.java:131)
        at org.apache.commons.logging.LogFactory.getLog(LogFactory.java:351)

似乎这可能来自以下依赖项:

   <dependency>
        <groupId>commons-httpclient</groupId>
        <artifactId>commons-httpclient</artifactId>
        <version>3.1</version>
    </dependency>

因为日志记录直接有效,并且在遇到commons-httpclient内容时会抛出错误。

我已尝试向我的slf4j添加各种pom.xml依赖项,但没有任何帮助,并且它无助于我理解实际问题。

有没有办法只使用log4j而没有任何错误?如果没有,如何消除这个错误?

3 个答案:

答案 0 :(得分:1)

首先尝试检查哪些其他项目依赖项依赖于slf4j

mvn dependency:tree

[INFO] Scanning for projects...
[INFO]
[INFO] ------------------------------------------------------------------------
[INFO] Building MavenScrap 1.0-SNAPSHOT
[INFO] ------------------------------------------------------------------------
[INFO]
[INFO] --- maven-dependency-plugin:2.1:tree (default-cli) @ MavenScrap  ---
[INFO] sub.optimal:MavenScrap:jar:1.0-SNAPSHOT
[INFO] +- commons-httpclient:commons-httpclient:jar:3.1:compile
[INFO] |  +- commons-logging:commons-logging:jar:1.0.4:compile
[INFO] |  \- commons-codec:commons-codec:jar:1.2:compile
[INFO] \- log4j:log4j:jar:1.2.17:compile
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------

答案 1 :(得分:1)

使用

检查依赖关系树
 mvn dependency:tree

找出哪个依赖项具有传递性slf4j依赖项。我怀疑它是commons-httpclient,因为它只有以下依赖项(并使用commons日志记录):

[INFO] --- maven-dependency-plugin:2.10:tree (default-cli) @ log4j ---
[INFO] binding:log4j:jar:0.0.1-SNAPSHOT
[INFO] \- commons-httpclient:commons-httpclient:jar:3.1:compile
[INFO]    +- commons-logging:commons-logging:jar:1.0.4:compile
[INFO]    \- commons-codec:commons-codec:jar:1.2:compile

您最终必须找出使用的是哪个slf4j-api版本(请查看mvn dependency:resolve)并提供匹配的log4j绑定库(请参阅SLF4J Manual)。

    <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-log4j12</artifactId>
        <version>x.x.x</version>
    </dependency>

答案 2 :(得分:1)

我们需要了解一些差异。

对于图书馆制作者

在这种情况下,我们正在开发Java库,将来的目标是将其包含在项目中。因此,如果我们想上天堂,我们不应在库中使用专有的日志记录提供程序(logback,log4j,jdk日志记录)。我们可以认为使用我们库的项目可以免费使用任何记录程序提供程序。

对于这种情况,SLF4J帮助我们。

pom.xml

<dependency>
  <groupId>org.slf4j</groupId>
  <artifactId>slf4j-api</artifactId>
  <version>1.7.5</version>
</dependency>
<dependency>
  <groupId>org.slf4j</groupId>
  <artifactId>slf4j-simple</artifactId>
  <version>1.7.5</version>
</dependency>

sample.java

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
private final Logger logger = LoggerFactory.getLogger(getClass());
Logger.info("hello");

现在您将看到STDOUT上具有INFO级别的日志记录输出。这个简单的记录器将默认显示任何INFO级别或更高级别的消息。为了查看DEBUG消息,您需要在Java启动时将级别设置为“系统属性”(选择一个)

  • -Dorg.slf4j.simpleLogger.defaultLogLevel=DEBUG
  • System.setProperty("org.slf4j.simpleLogger.defaultLogLevel", "DEBUG");

对于应用程序制造商

在这种情况下,我们不开发库。因此,我们可以自由使用任何我们喜欢的记录器提供商。就我而言,我喜欢log4j2。

pom.xml

<dependency>
  <groupId>org.apache.logging.log4j</groupId>
  <artifactId>log4j-api</artifactId>
  <version>2.7</version>
</dependency>
<dependency>
  <groupId>org.apache.logging.log4j</groupId>
  <artifactId>log4j-core</artifactId>
  <version>2.7</version>
</dependency>
<dependency>
  <groupId>org.apache.logging.log4j</groupId>
  <artifactId>log4j-slf4j-impl</artifactId>
  <version>2.7</version>
</dependency>

src / main / resources / log4j2.xml

<?xml version="1.0" encoding="UTF-8"?>
<Configuration status="WARN">
  <Appenders>
    <Console name="Console" target="SYSTEM_OUT">
      <PatternLayout pattern="%d{HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n"/>
    </Console>
  </Appenders>
  <Loggers>
    <Root level="debug">
      <AppenderRef ref="Console"/>
    </Root>
  </Loggers>
</Configuration>

sample.java

import org.apache.logging.log4j.Logger;
import org.apache.logging.log4j.LogManager;
private final Logger logger = LogManager.getLogger(this.getClass());
Logger.info("hello");

通过这些最小的配置,您将使用log4j配置获得幸福而健康的日志。

如果导入的库使用SLF4J,您将在log4j的同一通道中看到其日志。

但是,如果您导入的不使用SLF4J的库,例如:logback或jdk-logging,您将无法在 log4j 的同一频道中看到它们的日志,这可能是一场噩梦


参考