java - Log4j 配置 - 不同文件的不同日志

标签 java logging log4j

对于某些人来说,这可能是一个非常简单的问题,但我个人认为 Log4j 配置非常困难,而且学习进行脑部手术可能不那么具有挑战性。

我正在尝试让多个记录器登录到不同的文件。 这是我的 log4j.properties 文件中的内容:

# Root logger option
log4j.rootLogger=INFO, file, admin

# Direct log messages to a log file
log4j.appender.file=org.apache.log4j.RollingFileAppender
log4j.appender.file.File=/home/nick/logging/file.log
log4j.appender.file.MaxFileSize=1MB
log4j.appender.file.MaxBackupIndex=1
log4j.appender.file.layout=org.apache.log4j.PatternLayout
log4j.appender.file.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} %-5p %c{1} - %m%n

log4j.appender.admin=org.apache.log4j.RollingFileAppender
log4j.appender.admin.File=/home/nick/logging/admin.log
log4j.appender.admin.MaxFileSize=1MB
log4j.appender.admin.MaxBackupIndex=1
log4j.appender.admin.layout=org.apache.log4j.PatternLayout
log4j.appender.admin.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} %-5p %c{1} - %m%n

这是我用来测试配置的(非常简单的)Java 应用程序:

public static void main(String[] args) throws Exception {

    Properties resource = new Properties();
    InputStream in = new FileInputStream("/home/nick/logging/log4j.properties");
    resource.load(in);
    PropertyConfigurator.configure(resource);

    Logger admin = Logger.getLogger("admin");
    Logger file = Logger.getLogger("file");

    admin.info("hello admin");
    file.info("hello file");
}

我有两个问题:

一个问题我总是在 PropertyConfigurator.configure(resource);:

行中遇到异常
java.io.FileNotFoundException: /home/nick/logging (Is a directory)
 at java.io.FileOutputStream.open(Native Method)
 at java.io.FileOutputStream.<init>(FileOutputStream.java:212)
 at java.io.FileOutputStream.<init>(FileOutputStream.java:136)
 at org.apache.log4j.FileAppender.setFile(FileAppender.java:289)
 at org.apache.log4j.RollingFileAppender.setFile(RollingFileAppender.java:167)
 at org.apache.log4j.FileAppender.activateOptions(FileAppender.java:163)
 at org.apache.log4j.config.PropertySetter.activate(PropertySetter.java:256)

第二个问题是两条消息都写入了两个日志。这是实际结果:

文件管理员:日志:

2014-04-27 11:55:30 INFO  admin - hello admin
2014-04-27 11:55:30 INFO  file - hello file

文件file.log:

2014-04-27 11:55:30 INFO  admin - hello admin
2014-04-27 11:55:30 INFO  file - hello file

这是必需的结果:

文件管理员:日志:

2014-04-27 11:55:30 INFO  admin - hello admin

文件file.log:

2014-04-27 11:55:30 INFO  file - hello file

是什么导致了异常,我怎样才能达到要求的结果?

最佳答案

Log4J 区分了 loggersappenders,它们负责生成日志消息,而 appenders 负责将这些消息发送到某处(文件、控制台、数据库等)。记录器形成一个层次结构,根记录器是名为 admin 的记录器的父级,它是 admin.component1 等的父级,您可以将附加程序附加到任何层次结构中的记录器。默认情况下,记录器将向直接附加到它的所有附加程序发送消息,或者向其层次结构中的任何祖先发送消息(这就是记录器通常被命名为 Java 类的原因,例如,您可以控制 com.example 的日志记录.Class1com.example.subpkg.AnotherClass 通过配置 com.example 记录器)。

Loggers 和 appender 形成不同的命名空间,这就是你混淆的根源 - 名为 admin 的 logger 和名为 admin 的 appender 是两个独立的实体。

您在问题中给出的配置定义了一个记录器(根记录器),它将它生成的所有消息发送到两个单独的 appenders,两个文件各一个。然后,您的代码请求两个不同的记录器,并为每个记录器生成一条日志消息。这两个 logger 都从根 logger 继承了 appender 配置,因此它们都将消息发送到 both 配置的 appender。

enter image description here

您应该将 file 附加器附加到 file 记录器并将 admin 附加器附加到根记录器,而不是将两个附加器附加到根记录器admin 记录器:

log4j.rootLogger=INFO
log4j.logger.file=INFO, file
log4j.logger.admin=INFO, admin

这样 file 记录器将只向 file.log 发送消息,admin 记录器只向 admin.log,并且所有来自 other 记录器的消息都将被静默丢弃,因为根没有附加任何附加程序。

enter image description here


additivity 标志是此规则的异常(exception) - 将记录器的可加性设置为 false 实质上断开了从记录器到其父级的箭头,因此该记录器生成的消息(或流入它的消息来自它的一个 child )不会再往上走,他们只会去附加到直接到相关记录器的附加程序。

关于java - Log4j 配置 - 不同文件的不同日志,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23322602/

相关文章:

Java在方法之间传递参数

java - PostgreSQL、pgAdmin、Java : How to make them all UTC?

java - log4j:WARN 在 Java 项目中找不到记录器的附加程序

java - log4j 记录器不可见

java - 如何在 spring boot 测试(集成测试)期间查看自动配置日志输出

java - 从映射列表中输出键/值对

java - 使用 Spring Security 处理静态资源

java - 我是否需要 SLF4J 绑定(bind)和 log4j 适配器?

java - 如何解决 Java 中的事务日志?

java - logger.info 和 logger.debug 之间的区别