我正在尝试在运行时手动重新配置log4j2,但获得了部分成功。
以下是相关代码:
package examples.test;
public class ABCImpl implements XYX{
static Logger logger;
public void initialize(){
LoggerContext ctx = null;
Configuration config = null;
Map mp = null;
ctx = (LoggerContext) LogManager.getContext(false);
config = (Configuration)ctx.getConfiguration();
mp = config.getAppenders();
System.out.println("***<Provider o/p follows> Before logger re-configuration:");
System.out.println("\tAppenders:" + mp.keySet());
//reconfiguration attempt - starts
try{
URI configuration = this.getClass().getResource("/log4j2.xml").toURI();
ctx = Configurator.initialize(this.getClass().getName(), null, configuration);
}catch(Exception e){
System.out.println("\t-------Exception encountered-------");
e.printStackTrace();
}
config = (Configuration) ctx.getConfiguration();
mp = config.getAppenders();
System.out.println("***<Provider o/p follows> After logger re-configuration:");
System.out.println("\tAppenders:" + mp.keySet());
//reconfiguration attempt - ends
logger = LogManager.getLogger(this.getClass().getName());
}
public void myBusinessMethod(){
logger.info("Entry..............");
logger.info("Exit..............");
}
}
这个类实际上是jar文件的一部分,我在应用程序服务器中运行它,保证在我的类实例化后立即调用我的initialize
方法。
这是我的log4j2.xml,我已将其打包到jar的root:
<?xml version="1.0" encoding="UTF-8"?>
<!-- Don't forget to set system property
-DLog4jContextSelector=org.apache.logging.log4j.core.async.AsyncLoggerContextSelector
to make all loggers asynchronous. -->
<Configuration status="info">
<Appenders>
<!-- Async Loggers will auto-flush in batches, so switch off immediateFlush. -->
<RollingRandomAccessFile name="Appender1" fileName="servers/${sys:weblogic.Name}/logs/Auditing_${sys:weblogic.Name}.log" immediateFlush="true" append="false" filePattern="servers/${sys:weblogic.Name}/logs/archive/Auditing_${sys:weblogic.Name}-%d{yyyy-MM-dd-HH}-%i.log.gz">
<PatternLayout>
<Pattern>%d %p %c{1.} [%t] %m %ex%n</Pattern>
</PatternLayout>
<Policies>
<TimeBasedTriggeringPolicy interval="1" modulate="true"/>
<SizeBasedTriggeringPolicy size="250 MB"/>
</Policies>
<DefaultRolloverStrategy max="20"/>
</RollingRandomAccessFile>
<Async name="Async1">
<AppenderRef ref="Appender1"/>
</Async>
</Appenders>
<Loggers>
<Logger name="examples.test.ABCImpl" level="info" includeLocation="false" additivity="false">
<AppenderRef ref="Appender1"/>
</Logger>
<Root level="info" includeLocation="false">
<AppenderRef ref="Appender1"/>
</Root>
</Loggers>
</Configuration>
问题是即使创建了日志文件,但也没有记录任何内容。我得到的标准输出是:
ABCImpl.initialize
activeHandlerEntries.length=1
***<Provider o/p follows> Before logger re-configuration:
Appenders:[Console]
***<Provider o/p follows> After logger re-configuration:
Appenders:[Async1, Appender1]
只是为了确保我的log4j2.xml
和一段代码没有运行时重新配置的东西,如果我使用-Dlog4j.configurationFile=file:SOME_PATH_OUTSIDE_JAR/log4j2.xml
日志按预期填充。
请帮忙。
答案 0 :(得分:1)
几个小时的无助盯着代码并在这里和那里尝试我的手,我发现如果我从新初始化的上下文(从Logger
获得)而不是{{1}获得Configurator.initialize
用日志填充数据。
如果没有这个技巧,我不确定是否需要调用任何其他API方法(仅在LogManager
之后)将新上下文挂钩到Configurator.initialize
,以便我可以继续传统方法获取{{ 1}}来自LogManager
。
Logger
答案 1 :(得分:1)
我想知道,如果通过
派生初始化的上下文
ctx = (LoggerContext) LogManager.getContext(false);
可能是问题,因为你必须访问&#34;当前&#34;上下文
ctx = (LoggerContext) LogManager.getContext(true);
根据the Log4j2 API。我非常感谢澄清,因为我也尝试以编程方式配置Log4J2。