Java 如何在log4j中创建我自己的Appender?

Java 如何在log4j中创建我自己的Appender?,java,log4j,appender,Java,Log4j,Appender,我是log4j的新手。有人能解释一下如何创建我自己的Appender吗?i、 e.如何实现类和接口以及如何覆盖它?您应该扩展AppenderSkleton类,该类(引用javadoc)“提供通用功能的代码,例如支持阈值筛选和支持常规筛选。” 如果您阅读AppenderSkeleton的代码,您会发现它几乎可以处理所有问题,只留给您: 受保护的无效附加(LoggingEvent事件) 公众假期结束() 公共布尔值requiresLayout() 核心方法是append。请记住,您不需要在其中实现过

我是log4j的新手。有人能解释一下如何创建我自己的Appender吗?i、 e.如何实现类和接口以及如何覆盖它?

您应该扩展AppenderSkleton类,该类(引用javadoc)“提供通用功能的代码,例如支持阈值筛选和支持常规筛选。”

如果您阅读AppenderSkeleton的代码,您会发现它几乎可以处理所有问题,只留给您:

  • 受保护的无效附加(LoggingEvent事件)
  • 公众假期结束()
  • 公共布尔值requiresLayout()
  • 核心方法是append。请记住,您不需要在其中实现过滤逻辑,因为它已经在doAppend中实现,而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;
        }
    
    }
    

    你应该有“第一”,“第二”,“第三”打印;不应打印第四条消息,因为根记录器的日志级别为debug,而事件级别为trace。这证明AbstractSkeleton为我们正确地实现了“级别管理”。所以这似乎是最好的选择。。。现在的问题是:为什么您需要一个定制的appender,而该日志中几乎有许多内置到任何目的地的appender?(顺便说一句,最好从log4j开始:)

    要创建自己的Appender,只需实现Appender接口并覆盖它即可。
    同时也要研究这个链接开始

    如果你想做一些操作或决定,你可以这样做:

    @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 answer来支持pro文件配置以及启动和停止日志捕获的能力:

    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();
    

    您需要实现。但是您可能应该对现有的一个appender子类化,并重写方法,例如doAppend(LoggingEvent),以满足您的需要。如果现有appender是final呢?我使用了上面的类,没有错误,。。如何识别将创建的新Appender?如何使用它?如果p>=q,则在具有(分配或继承的,适当的)级别q的记录器中启用级别p的日志请求。日志请求的级别为“跟踪”,而记录器的级别设置为“调试”“它高于跟踪,因此不会被记录。请记住,记录器没有指定级别,因此它继承根记录器的级别。要让日志记录器日志也跟踪级别请求(最大冗余!),您可以通过编程方式(l.setLevel(level.trace);)或(更好)通过属性文件设置“logger”或根日志记录器的级别。我不知道您的特定web服务器。您是否尝试过“Tomcat下的默认初始化”中的建议?这不是您在最初的问题中提出的问题。如果您已经创建了自己的Appender子类,那么应该提供用于写入磁盘的代码。可能您不需要自定义Appender类,您只需要一个FileAppender,即开箱即用地登录到磁盘上。您只需提供文件名。@Elakkiya:不客气。您解决了log4j的问题了吗?您是否扩展了AppenderSkleton或使用了一些提供的Appender?错误:从以下内容中删除“StringBuffer”:StringBuffer logs=new StringBuffer()@LennarTroller,它不是一个bug。我在重写append(LoggingEvent事件)方法时出错-
    MyCustomAppender类型的方法append(LoggingEvent)必须重写或实现超类型方法
    StringBuffer sb = appender.stop();