对于某些人来说,这可能是一个非常简单的问题,但我个人认为 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 区分了 loggers 和 appenders,它们负责生成日志消息,而 appenders 负责将这些消息发送到某处(文件、控制台、数据库等)。记录器形成一个层次结构,根记录器是名为 admin
的记录器的父级,它是 admin.component1
等的父级,您可以将附加程序附加到任何层次结构中的记录器。默认情况下,记录器将向直接附加到它的所有附加程序发送消息,或者向其层次结构中的任何祖先发送消息(这就是记录器通常被命名为 Java 类的原因,例如,您可以控制 com.example 的日志记录.Class1
和 com.example.subpkg.AnotherClass
通过配置 com.example
记录器)。
Loggers 和 appender 形成不同的命名空间,这就是你混淆的根源 - 名为 admin
的 logger 和名为 admin
的 appender 是两个独立的实体。
您在问题中给出的配置定义了一个记录器(根记录器),它将它生成的所有消息发送到两个单独的 appenders,两个文件各一个。然后,您的代码请求两个不同的记录器,并为每个记录器生成一条日志消息。这两个 logger 都从根 logger 继承了 appender 配置,因此它们都将消息发送到 both 配置的 appender。
您应该将 file
附加器附加到 file
记录器并将 admin
附加器附加到根记录器,而不是将两个附加器附加到根记录器admin
记录器:
log4j.rootLogger=INFO
log4j.logger.file=INFO, file
log4j.logger.admin=INFO, admin
这样 file
记录器将只向 file.log
发送消息,admin
记录器只向 admin.log
,并且所有来自 other 记录器的消息都将被静默丢弃,因为根没有附加任何附加程序。
additivity 标志是此规则的异常(exception) - 将记录器的可加性设置为 false 实质上断开了从记录器到其父级的箭头,因此该记录器生成的消息(或流入它的消息来自它的一个 child )不会再往上走,他们只会去附加到直接到相关记录器的附加程序。
关于java - Log4j 配置 - 不同文件的不同日志,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23322602/