zoukankan      html  css  js  c++  java
  • 使用最新的log4cplus(1.1.1)隔离不同的 log 文件输出

    部分参考了博客。

    http://www.cppblog.com/tx7do/articles/11719.html


    基于脚本配置来过滤log信息
    除了通过程序实现对log环境的配置之外,log4cplus通过PropertyConfigurator类实现了基于脚本配置的功能。通过
    脚本可以完成对logger、appender和layout的配置,因此可以解决怎样输出,输出到哪里的问题,我将在全文的最后
    一部分中提到多线程环境中如何利用脚本配置来配合实现性能测试,本节将重点介绍基脚本实现过滤log信息的功能。

    首先简单介绍一下脚本的语法规则:
    包括Appender的配置语法和logger的配置语法,其中:

    1.Appender 的配置语法:
    1.1 设置名称:
    /*设置方法*/log4cplus.appender.appenderName=fully.qualified.name.of.appender.class
    例如(列举了所有可能的Appender,其中SocketAppender这里没有使用):
    log4cplus.appender.append_1=log4cplus::ConsoleAppender
    log4cplus.appender.append_2=log4cplus::FileAppender
    log4cplus.appender.append_3=log4cplus::RollingFileAppender
    log4cplus.appender.append_4=log4cplus::DailyRollingFileAppender
    log4cplus.appender.append_4=log4cplus::SocketAppender

    1.2. 设置Filter:
    包括选择过滤器和设置过滤条件,可选择的过滤器包括:LogLevelMatchFilter、LogLevelRangeFilter、和StringMatchFilter:
    对LogLevelMatchFilter来说,过滤条件包括LogLevelToMatch和AcceptOnMatch(true|false), 只有当log信息的LogLevel值与LogLevelToMatch相同,且AcceptOnMatch为true时才会匹配。
    LogLevelRangeFilter来说,过滤条件包括LogLevelMin、LogLevelMax和AcceptOnMatch,只有当log信息的LogLevel在LogLevelMin、LogLevelMax之间同时AcceptOnMatch为true时才会匹配。
    对StringMatchFilter来说,过滤条件包括StringToMatch和AcceptOnMatch,只有当log信息的LogLevel值与StringToMatch对应的LogLevel值与相同, 且AcceptOnMatch为true时会匹配。
        过滤条件处理机制类似于IPTABLE的Responsibility chain,(即先deny、再allow)不过执行顺序刚好相反,后写的条件会被先执行,比如:
    log4cplus.appender.append_1.filters.1=log4cplus::spi::LogLevelMatchFilterlog4cplus.appender.append_1.filters.1.LogLevelToMatch=TRACElog4cplus.appender.append_1.filters.1.AcceptOnMatch=true#log4cplus.appender.append_1.filters.2=log4cplus::spi::DenyAllFilter
    会首先执行filters.2的过滤条件,关闭所有过滤器,然后执行filters.1,仅匹配TRACE信息。

    1.3. 设置Layout
    可以选择不设置、TTCCLayout、或PatternLayout
    如果不设置,会输出简单格式的log信息。
    设置TTCCLayout如下所示:log4cplus.appender.ALL_MSGS.layout=log4cplus::TTCCLayout
    设置PatternLayout如下所示:log4cplus.appender.append_1.layout=log4cplus::PatternLayoutlog4cplus.appender.append_1.layout.ConversionPattern=%d{%m/%d/%y %H:%M:%S,%Q} [%t] %-5p - %m%n

    2.logger的配置语法
    同一个 logger 下的 Appender 会输出内容到该logger 下的所有文件,可以通过 LogLevel 等措施拉过滤。
    下面演示了建立不同logger,隔离输出内容的方法。
    包括rootLogger和non-root logger。
    对于rootLogger来说:log4cplus.rootLogger=[LogLevel], appenderName, appenderName, ...
    对于non-root logger来说:log4cplus.logger.logger_name=[LogLevel|INHERITED], appenderName, appenderName, ...
        脚本方式使用起来非常简单,只要首先加载配置即可(urconfig.properties是自行定义的配置文件):
    PropertyConfigurator::doConfigure("urconfig.properties");下面我们通过例子体会一下log4cplus强大的基于脚本过滤log信息的功能。


    下面建立的是 VS2012 的WIN32控制台工程 log4cplus_test,用来演示日志输出,
    工程需要注意两点:
    1. 使用的是最新的 log4cplus-1.1.1 版本,链接的是其中的静态库 log4cplusSD.lib
    2. 工程需要设置 字符集为 "使用多字节字符集",设置方法是VS2012 菜单:
    项目->log4cplus_test属性->配置属性->字符集


    下面是配置文件 urconfig.properties 的内容,使用配置来控制log4cplus 的log 输出.

    #全局默认根 logger,这里忽略
    #log4cplus.rootLogger=TRACE, ALL_MSGS, TRACE_MSGS, DEBUG_INFO_MSGS, FATAL_MSGS

    #log4cplus.rootLogger=TRACE,ALL_MSGS
    #log4cplus.appender.ALL_MSGS=log4cplus::RollingFileAppender
    #log4cplus.appender.ALL_MSGS.File=./logout/all_msgs.log
    #log4cplus.appender.ALL_MSGS.layout=log4cplus::TTCCLayout


    #独立的 logger 的配置语法,支持两个 appender
    log4cplus.logger.APPfilelogger = TRACE,APP,APP_DAILY
    og4cplus.additivity.APPfilelogger = false


    #独立的 logger 的配置语法,同一个 logger 下会发送到所有文件,
    #是否写入到所有文件,通过 LogLevel 来控制
    log4cplus.logger.SYSfilelogger = TRACE,SYS
    #log4cplus.additivity.SYSfilelogger = TRUE

    #独立的 logger 的配置语法
    log4cplus.logger.ACCfilelogger = TRACE,ACC
    #log4cplus.additivity.ACCfilelogger = TRUE


    #支持只写入同一个 logger 下的指定文件
    log4cplus.appender.APP=log4cplus::RollingFileAppender
    log4cplus.appender.APP.File=./logout/app_msgs.log
    log4cplus.appender.APP.ImmediateFlush=false
    log4cplus.appender.APP.MaxFileSize=1MB
    #log4cplus.appender.APP.MinFileSize=1M
    log4cplus.appender.APP.MaxBackupIndex=3
    log4cplus.appender.APP.layout=log4cplus::PatternLayout
    log4cplus.appender.APP.layout.ConversionPattern=%D{%Y-%m-%d %H:%M:%S.%Q}|%-5p|%c[2]|%t|%F:%L|%m%n
    log4cplus.appender.APP.filters.1=log4cplus::spi::LogLevelRangeFilter
    log4cplus.appender.APP.filters.1.LogLevelMin=TRACE
    log4cplus.appender.APP.filters.1.LogLevelMax=FATAL


    #支持只写入同一个 logger 下的指定文件
    log4cplus.appender.APP_DAILY=log4cplus::DailyRollingFileAppender
    log4cplus.appender.APP_DAILY.File=./logout/app_msgs_d.log
    #MONTHLY,WEEKLY,DAILY,TWICE_DAILY,HOURLY,MINUTELY
    log4cplus.appender.APP_DAILY.Schedule=MINUTELY
    log4cplus.appender.APP_DAILY.DatePattern='.'yyyy-MM-dd
    log4cplus.appender.APP_DAILY.ImmediateFlush=false
    log4cplus.appender.APP_DAILY.MaxBackupIndex=3
    log4cplus.appender.APP_DAILY.layout=log4cplus::PatternLayout
    log4cplus.appender.APP_DAILY.layout.ConversionPattern=%D{%Y-%m-%d %H:%M:%S.%Q}|%-5p|%c[2]|%t|%F:%L|%m%n
    log4cplus.appender.APP_DAILY.filters.1=log4cplus::spi::LogLevelRangeFilter
    log4cplus.appender.APP_DAILY.filters.1.LogLevelMin=TARCE
    log4cplus.appender.APP_DAILY.filters.1.LogLevelMax=FATAL


    #支持只写入同一个 logger 下的指定文件
    log4cplus.appender.SYS=log4cplus::RollingFileAppender
    log4cplus.appender.SYS.File=./logout/sys_msgs.log
    log4cplus.appender.SYS.MaxFileSize=1MB
    log4cplus.appender.SYS.MaxBackupIndex=3
    log4cplus.appender.SYS.ImmediateFlush=false
    log4cplus.appender.SYS.layout=log4cplus::PatternLayout
    log4cplus.appender.SYS.layout.ConversionPattern=%D{%Y-%m-%d %H:%M:%S.%Q}|%-5p|%c[2]|%t|%F:%L|%m%n
    log4cplus.appender.SYS.filters.1=log4cplus::spi::LogLevelRangeFilter
    log4cplus.appender.SYS.filters.1.LogLevelMin=TRACE
    log4cplus.appender.SYS.filters.1.LogLevelMax=FATAL

    #支持只写入同一个 logger 下的指定文件
    log4cplus.appender.ACC=log4cplus::RollingFileAppender
    log4cplus.appender.ACC.File=./logout/acc_msgs.log
    log4cplus.appender.ACC.MaxFileSize=1MB
    log4cplus.appender.ACC.MaxBackupIndex=3
    log4cplus.appender.ACC.ImmediateFlush=false
    log4cplus.appender.ACC.layout=log4cplus::PatternLayout
    log4cplus.appender.ACC.layout.ConversionPattern=%D{%Y-%m-%d %H:%M:%S.%Q}|%-5p|%c[2]|%t|%F:%L|%m%n
    log4cplus.appender.ACC.filters.1=log4cplus::spi::LogLevelRangeFilter
    log4cplus.appender.ACC.filters.1.LogLevelMin=TRACE
    log4cplus.appender.ACC.filters.1.LogLevelMax=FATAL


    主要代码文件是 log4cplus_test.cpp,代码如下:

    // log4cplus_test.cpp : 定义控制台应用程序的入口点。
    //

    #include "stdafx.h"
    #include <conio.h>
    #include <iostream>
    #include <xmemory>
    #include <log4cplus/loggingmacros.h>
    #include <log4cplus/appender.h>
    #include <log4cplus/logger.h>
    #include <log4cplus/fileappender.h>
    #include <log4cplus/consoleappender.h>
    #include <log4cplus/configurator.h>
    #include <log4cplus/helpers/property.h>

    using namespace std;
    using namespace log4cplus;

    //APP 和 APP_DAILY 共用了一个 logger配置
    static Logger APP = Logger::getInstance(LOG4CPLUS_TEXT("APPfilelogger.APP"));
    static Logger APP_DAILY = Logger::getInstance(LOG4CPLUS_TEXT("APPfilelogger.APP_DAILY"));

    //SYS,ACC 是独立的 logger 配置,这样 SYS,ACC,APP(APP_DAILY) 分别属于不同的logger
    //它们可以独立的输出文件,相互不干扰
    static Logger SYS = Logger::getInstance(LOG4CPLUS_TEXT("SYSfilelogger.SYS"));
    static Logger ACC = Logger::getInstance(LOG4CPLUS_TEXT("ACCfilelogger.ACC"));


    void printDebug()
    {   
        LOG4CPLUS_TRACE(APP, "::printDebug()");   
        LOG4CPLUS_DEBUG(APP, "This is a DEBUG message");
        LOG4CPLUS_DEBUG(APP_DAILY, "This is a DEBUG message");

        LOG4CPLUS_INFO(ACC, "This is a INFO message");
        LOG4CPLUS_INFO(ACC, "This is a INFO message");   
        LOG4CPLUS_WARN(ACC, "This is a WARN message");

        LOG4CPLUS_ERROR(SYS, "This is a ERROR message");   
        LOG4CPLUS_FATAL(SYS, "This is a FATAL message");
        LOG4CPLUS_TRACE(SYS, "::printDebug()");
    }

    void ta()
    {
        log4cplus::DailyRollingFileAppender *_append;
        _append = new DailyRollingFileAppender("./logout/dd.log",MINUTELY,false,5);
        _append->setName("file test");
        _append->setLayout(std::auto_ptr<TTCCLayout>(new TTCCLayout()));
        Logger::getRoot().addAppender(log4cplus::SharedAppenderPtr(_append));
        Logger test = Logger::getInstance("test");
        for(int i = 0 ;i < 10000000 ;++i)
        {
            LOG4CPLUS_TRACE(test, "::printDebug()");
            LOG4CPLUS_FATAL(test, "This is a FATAL message");
            LOG4CPLUS_FATAL(test, "This is a FATAL message");
            LOG4CPLUS_FATAL(test, "This is a FATAL message");
            LOG4CPLUS_FATAL(test, "This is a FATAL message");
        }
    }

    int _tmain(int argc, _TCHAR* argv[])
    {
        PropertyConfigurator config(LOG4CPLUS_TEXT("./urconfig.properties"));
        config.configure();

        for(int i = 0; i< 1000000;++i)
        {
            printDebug();
        }

        ::system("pause");

        return 0;
    }

  • 相关阅读:
    Photoshop
    你会为了钱出售自己的个人资料吗?
    [ElasticSearch] 空间搜索 (一)
    hdu1584 A strange lift (电梯最短路径问题)
    Android API Guides---OpenGL ES
    Qt 推断一个IP地址是否有效
    bzoj1670【Usaco2006 Oct】Building the Moat 护城河的挖掘
    集成学习1-Boosting
    微信开发模式之自己定义菜单实现
    人件札记:开放式的办公室环境
  • 原文地址:https://www.cnblogs.com/keanuyaoo/p/3299518.html
Copyright © 2011-2022 走看看