我们使用的是log4j 2.5。我的问题是关于ConferenceLog,所以请忽略下面的其他配置。
log4j2.xml文件:
<?xml version="1.0" encoding="UTF-8"?>
<Configuration monitorInterval="180">
<Properties>
<!-- change the path to conference logs below to fit your system -->
<Property name="LOG_DIR">C:CodeReposlogs</Property>
<Property name="PATTERN">%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level
%logger{36} - %msg%n</Property>
</Properties>
<Appenders>
<Console name="Console" target="SYSTEM_OUT">
<PatternLayout pattern="${PATTERN}" />
</Console>
<Routing name="DailyLog">
<Routes pattern="${ctx:logFileName}">
<Route>
<RollingFile name="Rolling-${ctx:logFileName}"
fileName="${LOG_DIR}/symproxy.log"
filePattern="${LOG_DIR}/symproxy-%i.log">
<PatternLayout pattern="${PATTERN}" />
<SizeBasedTriggeringPolicy size="20971520" />
<DefaultRolloverStrategy max="100" />
</RollingFile>
</Route>
</Routes>
</Routing>
<Routing name="ConferenceLog">
<Routes pattern="${ctx:logFileName}">
<Route>
<RollingFile name="Rolling-${ctx:logFileName}"
fileName="${LOG_DIR}/conferences/${ctx:logFileName}.log"
filePattern="${LOG_DIR}/conferences/${ctx:logFileName}.%i.log.gz">
<PatternLayout pattern="${PATTERN}" />
<SizeBasedTriggeringPolicy size="20971520" />
</RollingFile>
</Route>
</Routes>
</Routing>
<Routing name="DeviceLog">
<Routes pattern="${ctx:logFileName}">
<Route>
<RollingFile name="Rolling-${ctx:logFileName}"
fileName="${LOG_DIR}/devices/${ctx:logFileName}.log"
filePattern="${LOG_DIR}/devices/${ctx:logFileName}.%i.log.gz">
<PatternLayout pattern="${PATTERN}" />
<SizeBasedTriggeringPolicy size="5kb" />
<DefaultRolloverStrategy max="3" />
</RollingFile>
</Route>
</Routes>
</Routing>
</Appenders>
<Loggers>
<Root level="info">
<AppenderRef ref="Console" />
<AppenderRef ref="DailyLog" />
</Root>
<Logger name="conference.logger" level="debug" additivity="false">
<AppenderRef ref="ConferenceLog" />
<AppenderRef ref="Console" />
</Logger>
<Logger name="device.logger" level="info" additivity="false">
</Logger>
</Loggers>
</Configuration>
我有一个小包装,在那里我实例化了这个会议记录器
private final Log conferenceLogger = LogFactory.getLog("conference.logger");
之后,它像一样使用
private void logger(Long meetingId, Log log, String className, Level level, String message, Throwable error) {
ThreadContext.put(LOG_FILE_NAME, String.valueOf(meetingId));
logger(conferenceLogger, level, className + "." + message, error);
ThreadContext.remove(LOG_FILE_NAME);
}
private static void logger(Log log, Level level, String message, Throwable error) {
if (level == Level.DEBUG) {
log.debug(message);
} else if (level == Level.ERROR) {
if (error == null) {
log.error(message);
} else {
log.error(message, error);
}
} else if (level == Level.FATAL) {
log.fatal(message);
} else if (level == Level.TRACE) {
log.trace(message);
} else if (level == Level.INFO) {
log.info(message);
} else if (level == Level.WARN) {
log.warn(message);
}
}
这将创建一组名为"会议id"的文件。然而,我的问题是,这些文件在打开后永远不会关闭。
我读过这篇类似的SO文章,但我没有引用我的FileAppender
来调用close,也没有Logger
对象,所以我不知道如何实现它的修复。如何以编程方式触发关闭这些文件?因此,我们的一台服务器出现了"打开的文件太多"错误。
在阅读了更多的文档后,我似乎需要IdlePurgePolicy,但我认为它对打开的文件数量没有任何影响。如果我的log4j2文件有几个不同的地方,我尝试了下面的行,但正如我所提到的,它对没有影响
<IdlePurgePolicy timeToLive="1" timeUnit="minutes"/>
如果包装器本身没有静态实例化,则会为调用它的每个对象创建记录器实例。这可能会导致每次都有一个新的套接字连接到具体的记录器实例。但是,当您只有一个日志文件时,打开的套接字过多也可能导致此异常。
可能的解决方案
- 静态实例化包装
- 静态实例化记录器引用
- 使包装器成为具有日志记录方法的静态调用的单例
<Routing name="ConferenceLog">
<Routes pattern="${ctx:logFileName}">
<Route>
<RollingFile name="Rolling-${ctx:logFileName}"
fileName="${LOG_DIR}/conferences/${ctx:logFileName}.log"
filePattern="${LOG_DIR}/conferences/${ctx:logFileName}.%i.log.gz">
<PatternLayout pattern="${PATTERN}" />
<SizeBasedTriggeringPolicy size="20971520" />
</RollingFile>
</Route>
</Routes>
<IdlePurgePolicy timeToLive="30" timeUnit="seconds"/>
</Routing>