【发布时间】:2014-12-03 14:41:42
【问题描述】:
我们使用 log4j 1.2.x 登录我们的产品,并希望在不久的将来迁移到 log4j 2.x。我们实现的功能之一是在生成的每个新的翻转日志文件上记录系统信息和其他重要参数。我们在 log4j 1.2.x 中实现的方式是我们扩展了 log4j 的 RollingFileAppender 类并覆盖了 rollOver() 方法,下面是实现的部分 sn-p
@Override
public void rollOver() {
super.rollOver(); //We are not modifying it's default functionality but as soon as rollOver happens we apply our logic
//
// Logic to log required system properties and important parameters.
//
}
现在我们想迁移到 log4j2,我们正在寻找一种新的解决方案来实现相同的功能。但是当我看到 log4j2 的源代码时,它与旧的源代码有很大不同。 RollingFileAppender 类不包含 rollover() 方法,因为它已移至 RollingManagerhelper 并且也已设置为 private。
开发一个完整的新包并从 log4j2 扩展/实现一些抽象/帮助类是我们可能的解决方案之一,但这需要大量的编码/复制,因为我们不修改 RollingFileAppender 所做的,而是我们只需要对其进行小的扩展。有一个简单的解决方案吗?
更新
我根据答案中的建议创建了一个自定义查找,以下是我创建它的方式;
@Plugin(name = "property", category = StrLookup.CATEGORY)
public class CustomLookup extends AbstractLookup {
private static AtomicLong aLong = new AtomicLong(0);
@Override
public String lookup(LogEvent event, String key) {
if (aLong.getAndIncrement() == 0) {
return "this was first call";
}
if (key.equalsIgnoreCase("customKey")) {
return getCustomHeader();
} else {
return "non existing key";
}
}
private static String getCustomHeader() {
// Implementation of custom header
return "custom header string";
}}
但这并没有像上面提到的那样工作;这总是在标题中打印this was first call。我还尝试将断点放在第一个 if 条件上,我注意到它只被调用一次。所以我担心的是 customLookup 类只有在 log4j2 从 xml 配置初始化它的属性时才会在启动时被初始化。我不知道我还能如何实现这个自定义查找类。
更新 2
在上面的实现之后,我尝试了一些不同的方式,如下所示;
private static AtomicLong aLong = new AtomicLong(0);
@Override
public String lookup(LogEvent event, String key) {
return getCustomHeader(key);
}
private static String getCustomHeader(final String key) {
if (aLong.getAndIncrement() == 0) {
return "this was first call";
}
if (key.equalsIgnoreCase("customKey")) {
// Implementation for customKey
return "This is custom header";
} else {
return "non existing key";
}
}
但这也做同样的事情。 log4j2 在从其 xml 配置文件初始化时创建标头,然后使用内存中的标头。被覆盖的lookup() 方法的return 值不能动态更改,因为它仅在初始化期间被调用。任何更多的帮助将不胜感激。
【问题讨论】:
-
如果要使用运行时变量替换,请在 log4j2 属性文件中指定变量键时使用双 $$。