log4jは初めてです。誰かが自分のアペンダーを作成する方法を説明できますか?つまり、クラスとインターフェイスを実装する方法と、それをオーバーライドする方法は?
4 に答える
更新:提供されたソリューションはLog4J1.xで有効です。2.xバージョンをお探しの場合は、次の記事をご覧ください:log4j2でカスタムアペンダーを作成する方法
AppenderSkeletonクラスを拡張する必要があります。これは、(javadocを引用して)「しきい値フィルタリングのサポートや一般的なフィルターのサポートなど、一般的な機能のコードを提供します」。
AppenderSkeletonのコードを読むと、ほとんどすべてを処理していることがわかります。
- 保護されたvoidappend(LoggingEventイベント)
- public void close()
- public boolean requireLayout()
コアメソッドはappendです。フィルタリングロジックはすでにdoAppendに実装されており、appendを呼び出すため、フィルタリングロジックを実装する必要はありません。ここでは、デモのように、ログエントリをArrayListに格納する(まったく役に立たない)クラスを作成しました。
public /*static*/ class MyAppender extends AppenderSkeleton {
ArrayList<LoggingEvent> eventsList = new ArrayList();
@Override
protected void append(LoggingEvent event) {
eventsList.add(event);
}
public void close() {
}
public boolean requiresLayout() {
return false;
}
}
では、テストしてみましょう。
public static void main (String [] args) {
Logger l = Logger.getLogger("test");
MyAppender app = new MyAppender();
l.addAppender(app);
l.warn("first");
l.warn("second");
l.warn("third");
l.trace("fourth shouldn't be printed");
for (LoggingEvent le: app.eventsList) {
System.out.println("***" + le.getMessage());
}
}
「first」、「second」、「third」を印刷する必要があります。イベントレベルがトレースである間、ルートロガーのログレベルはデバッグであるため、4番目のメッセージは出力されません。これは、AbstractSkeletonが「レベル管理」を正しく実装していることを証明しています。だから、それは間違いなく進むべき道のようです...今の質問:ほとんどすべての宛先にそのログが組み込まれているのに、なぜカスタムアペンダーが必要なのですか?(ところで、log4jから始めるのに適した場所:http://logging.apache.org/log4j/1.2/manual.html)
いくつかの操作や決定を行いたい場合は、次のように行うことができます。
@Override
protected void append(LoggingEvent event) {
String message = null;
if(event.locationInformationExists()){
StringBuilder formatedMessage = new StringBuilder();
formatedMessage.append(event.getLocationInformation().getClassName());
formatedMessage.append(".");
formatedMessage.append(event.getLocationInformation().getMethodName());
formatedMessage.append(":");
formatedMessage.append(event.getLocationInformation().getLineNumber());
formatedMessage.append(" - ");
formatedMessage.append(event.getMessage().toString());
message = formatedMessage.toString();
}else{
message = event.getMessage().toString();
}
switch(event.getLevel().toInt()){
case Level.INFO_INT:
//your decision
break;
case Level.DEBUG_INT:
//your decision
break;
case Level.ERROR_INT:
//your decision
break;
case Level.WARN_INT:
//your decision
break;
case Level.TRACE_INT:
//your decision
break;
default:
//your decision
break;
}
}
@AgostinoXの回答を使用して、プロファイル構成とロギングキャプチャを開始および停止する機能をサポートしたいと思います。
public class StringBufferAppender extends org.apache.log4j.AppenderSkeleton {
StringBuffer logs = new StringBuffer();
AtomicBoolean captureMode = new AtomicBoolean(false);
public void close() {
// TODO Auto-generated method stub
}
public boolean requiresLayout() {
// TODO Auto-generated method stub
return false;
}
@Override
protected void append(LoggingEvent event) {
if(captureMode.get())
logs.append(event.getMessage());
}
public void start()
{
//System.out.println("[StringBufferAppender|start] - Start capturing logs");
StringBuffer logs = new StringBuffer();
captureMode.set(true);
}
public StringBuffer stop()
{
//System.out.println("[StringBufferAppender|start] - Stop capturing logs");
captureMode.set(false);
StringBuffer data = new StringBuffer(logs);
logs = null;
return data;
}
}
これで、log4j.propertyファイルで定義するだけで済みます。
log4j.rootLogger=...., myAppender # here you adding your appendr name
log4j.appender.myAppender=com.roi.log.StringBufferAppender # pointing it to the implementation
runtume中に有効にしたいときよりも:
Logger logger = Logger.getRootLogger();
StringBufferAppender appender = (StringBufferAppender)logger.getAppender("myAppender");
appender.start();
そしてそれを止めたい間:
StringBuffer sb = appender.stop();
独自のアペンダーを作成するには、アペンダーインターフェイスを実装し、それをオーバーライドするだけです。また、このリンク開始ログを調べてください