使用log4cxx在GUI 程序中将信息输出到Console

之前看到有个方法是在项目属性设置里实现的

以VS2010为例:

右键Project选择Properties->Configuration Properties->Build Events->Post-Build Event,在Command Line后面添加

  editbin /SUBSYSTEM:CONSOLE $(OUTDIR)\$(TargetName).exe

该文同时指出“使用AllocConsole()的方法,对printf和cout有效,而对log4cxx无效”。

此法虽然可行,但是无论是否有信息输出到Console,程序启动就会开启一个Console窗口。不灵活。

我要补充的是:使用AllocConsole()方法也是可以的,只是调用的位置不在主程序中,而是对log4cxx的源码稍做修改即可。

因为是控制台log相关的,所以我决定修改consoleappender.h和consoleappender.cpp,要做的就是为程序分配一个Console窗口,红色部分为添加的代码

consoleappender.h

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#ifndef _LOG4CXX_CONSOLE_APPENDER_H
#define _LOG4CXX_CONSOLE_APPENDER_H

#include <windows.h>
#include <tchar.h>
#include <log4cxx/writerappender.h>

namespace log4cxx
{
 
        /**
        * ConsoleAppender appends log events to <code>stdout</code> or
        * <code>stderr</code> using a layout specified by the user. The
        * default target is <code>stdout</code>.
        */
        class LOG4CXX_EXPORT ConsoleAppender : public WriterAppender
        {
        private:
                void AllocConsole();
                LogString target;
                HWND m_wnd_console;

        public:
                DECLARE_LOG4CXX_OBJECT(ConsoleAppender)
                BEGIN_LOG4CXX_CAST_MAP()
                        LOG4CXX_CAST_ENTRY(ConsoleAppender)
                        LOG4CXX_CAST_ENTRY_CHAIN(AppenderSkeleton)
                END_LOG4CXX_CAST_MAP()

                ConsoleAppender();
                ConsoleAppender(const LayoutPtr& layout);
                ConsoleAppender(const LayoutPtr& layout, const LogString& target);
                ~ConsoleAppender();


                /**
                *  Sets the value of the <b>target</b> property. Recognized values
                *  are "System.out" and "System.err". Any other value will be
                *  ignored.
                * */
                void setTarget(const LogString& value);

                /**
                * Returns the current value of the <b>target</b> property. The
                * default value of the option is "System.out".
                *
                * See also #setTarget.
                * */
                LogString getTarget() const;

                void activateOptions(log4cxx::helpers::Pool& p);
                void setOption(const LogString& option, const LogString& value);
                static const LogString& getSystemOut();
                static const LogString& getSystemErr();


        private:
                void targetWarn(const LogString& val);
                static log4cxx::helpers::WriterPtr createWriter(const LogString& target);

        };
        LOG4CXX_PTR_DEF(ConsoleAppender);
}  //namespace log4cxx

#endif //_LOG4CXX_CONSOLE_APPENDER_H

consoleappender.cpp

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
#include <log4cxx/logstring.h>
#include <log4cxx/consoleappender.h>
#include <log4cxx/helpers/loglog.h>
#include <log4cxx/helpers/systemoutwriter.h>
#include <log4cxx/helpers/systemerrwriter.h>
#include <log4cxx/helpers/stringhelper.h>
#include <log4cxx/layout.h>

using namespace log4cxx;
using namespace log4cxx::helpers;

IMPLEMENT_LOG4CXX_OBJECT(ConsoleAppender)

ConsoleAppender::ConsoleAppender()
 : target(getSystemOut()), m_wnd_console(GetConsoleWindow())
{
    AllocConsole();
}

ConsoleAppender::ConsoleAppender(const LayoutPtr& layout1)
 :target(getSystemOut()), m_wnd_console(GetConsoleWindow())
{
    AllocConsole();
    setLayout(layout1);
    WriterPtr wr(createWriter(getSystemOut()));
    setWriter(wr);
    Pool p;
    WriterAppender::activateOptions(p);
}

ConsoleAppender::ConsoleAppender(const LayoutPtr& layout1, const LogString& target1)
    :m_wnd_console(GetConsoleWindow()), target(target1)
{
      AllocConsole();
      setLayout(layout1);
      WriterPtr wr(createWriter(target1));
      setWriter(wr);
      Pool p;
      WriterAppender::activateOptions(p);
}

ConsoleAppender::~ConsoleAppender()
{
        finalize();
}

const LogString& ConsoleAppender::getSystemOut() {
  static const LogString name(LOG4CXX_STR("System.out"));
  return name;
}

const LogString& ConsoleAppender::getSystemErr() {
  static const LogString name(LOG4CXX_STR("System.err"));
  return name;
}

WriterPtr ConsoleAppender::createWriter(const LogString& value) {
  LogString v = StringHelper::trim(value);

  if (StringHelper::equalsIgnoreCase(v,
         LOG4CXX_STR("SYSTEM.ERR"), LOG4CXX_STR("system.err"))) {
          return new SystemErrWriter();
  }
  return new SystemOutWriter();
}

void ConsoleAppender::setTarget(const LogString& value)
{
        LogString v = StringHelper::trim(value);

        if (StringHelper::equalsIgnoreCase(v,
              LOG4CXX_STR("SYSTEM.OUT"), LOG4CXX_STR("system.out")))
        {
                target = getSystemOut();
        }
        else if (StringHelper::equalsIgnoreCase(v,
               LOG4CXX_STR("SYSTEM.ERR"), LOG4CXX_STR("system.err")))
        {
                target = getSystemErr();
        }
        else
        {
                targetWarn(value);
        }
}

LogString ConsoleAppender::getTarget() const
{
        return target;
}

void ConsoleAppender::targetWarn(const LogString& val)
{
        LogLog::warn(((LogString) LOG4CXX_STR("["))
           + val +  LOG4CXX_STR("] should be system.out or system.err."));
        LogLog::warn(LOG4CXX_STR("Using previously set target, System.out by default."));
}

void ConsoleAppender::activateOptions(Pool& p)
{
        if(StringHelper::equalsIgnoreCase(target,
              LOG4CXX_STR("SYSTEM.OUT"), LOG4CXX_STR("system.out")))
        {
                WriterPtr writer1(new SystemOutWriter());
                setWriter(writer1);
        }
        else if (StringHelper::equalsIgnoreCase(target,
              LOG4CXX_STR("SYSTEM.ERR"), LOG4CXX_STR("system.err")))
        {
              WriterPtr writer1(new SystemErrWriter());
              setWriter(writer1);
        }
        WriterAppender::activateOptions(p);
}

void ConsoleAppender::setOption(const LogString& option, const LogString& value)
{
        if (StringHelper::equalsIgnoreCase(option,
              LOG4CXX_STR("TARGET"), LOG4CXX_STR("target")))
        {
                setTarget(value);
        }
        else
        {
                WriterAppender::setOption(option, value);
        }
}

void ConsoleAppender::AllocConsole() {
    if(NULL == m_wnd_console) {
        ::AllocConsole();
        m_wnd_console = GetConsoleWindow();
        FILE* stream = NULL;
        _tfreopen_s(&stream, _T("CONOUT$"), _T("w"), stdout);
    }
}

propertyconfigurator.cpp中修改如下

void PropertyConfigurator::doConfigure(const File& configFileName,
        spi::LoggerRepositoryPtr& hierarchy)
{
       std::locale::global(std::locale(""));
       hierarchy->setConfigured(true);

       Properties props;
       try {
          InputStreamPtr inputStream = new FileInputStream(configFileName);
          props.load(inputStream);
       } catch(const IOException& ie) {
          LogLog::error(((LogString) LOG4CXX_STR("Could not read configuration file ["))
                        + configFileName.getPath() + LOG4CXX_STR("]."));
          return;
       }

       try {
          doConfigure(props, hierarchy);
       } catch(const std::exception& ex) {
          LogLog::error(((LogString) LOG4CXX_STR("Could not parse configuration file ["))
                        + configFileName.getPath() + LOG4CXX_STR("]."), ex);
       }
}

 

在程序中写了一个对log4cxx的封装类CLoggerFramework,主要是为了修改控制台文本颜色和分离log(TRACE信息只输出到控制台,不输出到文件)

LoggerFramework.h

#pragma once
// log4cxx
#include "log4cxx/logger.h"
#include "log4cxx/propertyconfigurator.h"
using namespace log4cxx;

#define _TRACE     CLoggerFramework::LoggerInstance()->LoggerTrace
#define _INFO      CLoggerFramework::LoggerInstance()->LoggerInfo
#define _WARN      CLoggerFramework::LoggerInstance()->LoggerWarn
#define _ERROR     CLoggerFramework::LoggerInstance()->LoggerError

class CLoggerFramework {
public:
    // TODO: add your methods here.
    void LoggerTrace(const TCHAR* msg, ...);
    void LoggerInfo (const TCHAR* msg, ...);
    void LoggerWarn (const TCHAR* msg, ...);
    void LoggerError(const TCHAR* msg, ...);
    static CLoggerFramework* LoggerInstance();

private:
    CLoggerFramework(void);
    ~CLoggerFramework(void);
    void AttachConsoleAndSetTextColor(unsigned int color);

    unsigned int m_buf_len;
    static CLoggerFramework* m_pLoggerFramework;
    bool m_console_attached;
    // 之前字体颜色,如果相同则不用再设置
    unsigned int m_prev_color;
    // 控制台句柄,用来设置字体颜色
    HANDLE m_console_handle;
    CRITICAL_SECTION m_criticalSection;
    static LoggerPtr m_pFileLogger;
    static LoggerPtr m_pConsoleLogger;
};

LoggerFramework.cpp

#include "stdafx.h"
#include "LoggerFramework.h"

CLoggerFramework* CLoggerFramework::m_pLoggerFramework = NULL;
LoggerPtr CLoggerFramework::m_pFileLogger = NULL;
LoggerPtr CLoggerFramework::m_pConsoleLogger = NULL;

// This is the constructor of a class that has been exported.
// see LoggerFramework.h for the class definition
CLoggerFramework::CLoggerFramework() : m_buf_len(0) {
    m_console_attached = false;
    m_console_handle = NULL;
    m_prev_color = 0;
    InitializeCriticalSectionAndSpinCount(&m_criticalSection, 4000);
    // get logger
    PropertyConfigurator::configure(g_moduleDirectory + _T("log4cxx.properties"));
    m_pFileLogger = Logger::getLogger(_T("PMLog"));
    m_pConsoleLogger = Logger::getRootLogger();
}

CLoggerFramework::~CLoggerFramework() {
    DeleteCriticalSection(&m_criticalSection);
    SetConsoleTextAttribute(m_console_handle, FOREGROUND_RED | FOREGROUND_GREEN
        | FOREGROUND_BLUE);
}

CLoggerFramework* CLoggerFramework::LoggerInstance() {
    if(NULL == m_pLoggerFramework) {
        m_pLoggerFramework = new CLoggerFramework();
    }
    return m_pLoggerFramework;
}

void CLoggerFramework::LoggerTrace(const TCHAR* msg, ...) {
    EnterCriticalSection(&m_criticalSection);
    AttachConsoleAndSetTextColor(FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE);
    va_list argList;
    va_start(argList, msg);
    m_buf_len = _vsctprintf(msg, argList) + 1;
    TCHAR* buffer = new TCHAR[m_buf_len]();
    _vstprintf_s(buffer, m_buf_len, msg, argList);
    va_end(argList);
    std::tstring message(buffer);
    delete[] buffer;
    LOG4CXX_TRACE(m_pConsoleLogger, message.c_str());
    LeaveCriticalSection(&m_criticalSection);
}

void CLoggerFramework::LoggerInfo(const TCHAR* msg, ...) {
    EnterCriticalSection(&m_criticalSection);
    AttachConsoleAndSetTextColor(FOREGROUND_GREEN);
    va_list argList;
    va_start(argList, msg);
    m_buf_len = _vsctprintf(msg, argList) + 1;
    TCHAR* buffer = new TCHAR[m_buf_len]();
    _vstprintf_s(buffer, m_buf_len, msg, argList);
    va_end(argList);
    std::tstring message(buffer);
    delete[] buffer;
    LOG4CXX_INFO(m_pFileLogger, message.c_str());
    LeaveCriticalSection(&m_criticalSection);
}

void CLoggerFramework::LoggerWarn(const TCHAR* msg, ...) {
    EnterCriticalSection(&m_criticalSection);
    AttachConsoleAndSetTextColor(FOREGROUND_RED | FOREGROUND_GREEN);
    va_list argList;
    va_start(argList, msg);
    m_buf_len = _vsctprintf(msg, argList) + 1;
    TCHAR* buffer = new TCHAR[m_buf_len]();
    _vstprintf_s(buffer, m_buf_len, msg, argList);
    va_end(argList);
    std::tstring message(buffer);
    delete[] buffer;
    LOG4CXX_WARN(m_pFileLogger, message.c_str());
    LeaveCriticalSection(&m_criticalSection);
}

void CLoggerFramework::LoggerError(const TCHAR* msg, ...) {
    EnterCriticalSection(&m_criticalSection);
    AttachConsoleAndSetTextColor(FOREGROUND_RED);
    va_list argList;
    va_start(argList, msg);
    m_buf_len = _vsctprintf(msg, argList) + 1;
    TCHAR* buffer = new TCHAR[m_buf_len]();
    _vstprintf_s(buffer, m_buf_len, msg, argList);
    va_end(argList);
    std::tstring message(buffer);
    delete[] buffer;
    LOG4CXX_ERROR(m_pFileLogger, message.c_str());
    LeaveCriticalSection(&m_criticalSection);
}

void CLoggerFramework::AttachConsoleAndSetTextColor(unsigned int color) {
    if(!m_console_attached) {
        // 获取控制台句柄用来设置字体颜色
        m_console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
     SetConsoleTitle(_T("日志监控")); m_console_attached
= true; } if(color != m_prev_color) { SetConsoleTextAttribute(m_console_handle, color); m_prev_color = color; } }

配置文件如下

#设置root logger为TRACE级别,使用了ca和fa两个Appender
log4j.rootLogger = TRACE, ca
log4j.logger.PMLog = INFO, fa, ha

#对Appender ca进行设置
log4j.appender.ca = org.apache.log4j.ConsoleAppender
log4j.appender.ca.ImmediateFlush = true
log4j.appender.ca.Append = false
log4j.appender.ca.layout = org.apache.log4j.PatternLayout
log4j.appender.ca.layout.ConversionPattern = %d{yyyy-MM-dd HH:mm:ss,SSS}[%-5p]: %m%n

#对Appender fa进行设置
log4j.appender.fa = org.apache.log4j.FileAppender
log4j.appender.fa.ImmediateFlush = true
log4j.appender.fa.File = output.log
log4j.appender.fa.Append = false
log4j.appender.fa.MaxFileSize = 2MB
log4j.appender.fa.MaxBackupIndex = 10
log4j.appender.fa.layout = org.apache.log4j.PatternLayout
log4j.appender.fa.layout.ConversionPattern = %d{yyyy-MM-dd HH:mm:ss,SSS}[%t][%c][%-5p]: %m%n

#对Appender ha进行配置
log4j.appender.ha = org.apache.log4j.FileAppender
log4j.appender.ha.ImmediateFlush = true
log4j.appender.ha.File = output.htm
log4j.appender.ha.Append = false
log4j.appender.ha.MaxFileSize = 10MB
log4j.appender.ha.MaxBackupIndex = 10
log4j.appender.ha.layout = org.apache.log4j.HTMLLayout
log4j.appender.ha.layout.ConversionPattern = %d{yyyy-MM-dd HH:mm:ss,SSS}[%t][%c][%-5p]: %m%n

得到如下log效果

控制台:

文本log

htm网页log

另外关于每天产生一个日志文件的配置要注意,不要在File属性中指定文件名,否则不会按日期产生文件,要在DatePattern中指定,字符串最好用单引号分隔开,避免出现奇怪的文件后缀.还有,使用html格式输出时将layout指定为org.apache.log4j.HTMLLayout时layout.ConversionPattern指定的格式将失效.修改后的配置文件如下所示

#设置root logger为TRACE级别,使用了ca和fa两个Appender
log4j.rootLogger = TRACE, ca
log4j.logger.PMLog = INFO, fa, ha

#对Appender ca进行设置
log4j.appender.ca = org.apache.log4j.ConsoleAppender
log4j.appender.ca.ImmediateFlush = true
log4j.appender.ca.Append = false
log4j.appender.ca.layout = org.apache.log4j.PatternLayout
log4j.appender.ca.layout.ConversionPattern = [%-5p][%d{yyyy-MM-dd HH:mm:ss,SSS}]: %m%n

#对Appender fa进行设置
log4j.appender.fa = org.apache.log4j.DailyRollingFileAppender
log4j.appender.fa.ImmediateFlush = true
log4j.appender.fa.Append = true
#log4j.appender.fa.File = ./Logs/Text/ParkingLog.log
log4j.appender.fa.DatePattern = './Logs/Text/ParkingLog_'yyyy-MM-dd'.log'
log4j.appender.fa.layout = org.apache.log4j.PatternLayout
log4j.appender.fa.layout.ConversionPattern = [%-5p][%d{HH:mm:ss,SSS}]: %m%n

#对Appender ha进行配置
log4j.appender.ha = org.apache.log4j.DailyRollingFileAppender
log4j.appender.ha.ImmediateFlush = true
log4j.appender.ha.Append = true
#log4j.appender.ha.File = ./Logs/Htm/ParkingLog.htm
log4j.appender.ha.DatePattern = './Logs/Html/ParkingLog_'yyyy-MM-dd'.html'
log4j.appender.ha.layout = org.apache.log4j.HTMLLayout

 

 

posted @ 2014-06-05 15:21  琼'  阅读(1108)  评论(0编辑  收藏  举报