【转自】http://blog.csdn.net/ylioi/article/details/9049591

这篇文章对Log4cpp使用了宏定义和类进行封装,非常有借鉴意义。

log4cpp 是参考 log4j 所写的 c++ 版本的写 log 的库。可以在这里下载  http://log4cpp.sourceforge.net/
 
我的使用方法是:
1,定义了一个 _LOG4CPP 宏,用来打开或关闭 log4cpp 的调用,以便在完全不需要 log 的情况下可以运行,例如进行性能测试时;又比如 log4cpp 可能有内存泄露,关闭使用它,可以检查程序其它地方是否有泄露;
 
2,在写 log 时,自动加上函数名,以方便跟踪运行信息;(在 gcc 可以用可变参数宏定义,vc 中用其它办法,稍后说明)
 
3,每一个需要写 log 的类,都加上一个静态 log 实例,以方便通过配置,使得相应的 log 可以输出到指定的地方,有利于跟踪特定的运行信息;(如果不在配置文件中指定对应的Category, 则会默认为其生成一个NOTSET级别的Category, 因此对应日志会自动记录到 父Category指定的日志文件中)
 
4,不直接调用 log4cpp ,而是通过宏定义来使用,原因同1,即有利于关闭 log 功能;
 
基于以上几点,我写了几个函数及宏定义等,以下逐一说明:
 
// InitializeLog4cpp 从指定的 log 配置文件中读取信息,并初始化 log4cpp,这个函数在进程的入口处调用
void InitializeLog4cpp(const std::string & logfile);
 
// ReleaseLog4cpp 用于释入 log4cpp ,在进程的结束处调用
void ReleaseLog4cpp();
 
// 以下宏用于在类中声明一个静态变量,变量名为 "log"
DECLARE_STATIC_LOG()
 
// 以下宏用于初如化类中的 log 静态变量
DEFINE_STATIC_LOG(ClassName)
 
// 以下是用来写 log 信息的几个宏,严重级别分别为 DEBUG, INFO, NOTICE, ERROR
LogDebug
LogInfo
LogNotice
LogError
 
以下先演示一个应用实例:
 
Demo.h 文件主要内容:

#include "LogUtils.h"

using namespace dk;
class Demo
{
public:
Demo();
virtual ~Demo(); public:
void TestIt(int i, int j); private:
DECLARE_STATIC_LOG(); // 静态成员变量声明 };

Demo.cpp 文件主要内容:

#include "Demo.h"

DEFINE_STATIC_LOG(Demo);  // 静态成员变量初始化

Demo::Demo()
{
LogInfo("");
} Demo::~Demo()
{
LogInfo("");
} void Demo::TestIt(int i, int j)
{
LogInfo("i=%d, j=%d", i, j); // 支持多参数!
}

再写个例子来运行,可以看到类似输出

[INFO] - Demo::Demo() - 
[INFO] - Demo::TestIt() - i=1, j=2
[INFO] - Demo::~Demo() -

最后附上完整的

LogUtils.h

LogUtils.cpp

LogTracer.h (这个文件是因为 vc 不支持可变参数的宏,所以采用的取巧方法,这方法是从网上搜来的,出处忘了,请见谅。)

LogUtils.h

#ifndef _LOG_UTILS_H_
#define _LOG_UTILS_H_ #include <string>
#include "LogTracer.h" #if defined(_LOG4CPP) #include <log4cpp/Category.hh> namespace log4cpp
{
class Category;
}; #endif namespace dk
{
// 从指定的配置文件logfile中读取配置信息,并初始化log4cpp,这个函数在进程的入口处调用
void InitializeLog4cpp(const std::string & logfile);
void ReleaseLog4cpp(); // 用于释入 log4cpp ,在进程的结束处调用 #if defined (_LOG4CPP)
log4cpp::Category & GetLogCategory(const char * categoryName); // 配置文件中需包含名为 dk.categoryName的 category
#endif #if defined (_LOG4CPP)
# define DECLARE_STATIC_LOG() static log4cpp::Category & log // 用于初始化类中的 log 静态变量
# define DEFINE_STATIC_LOG(ClassName) log4cpp::Category & ClassName::log = GetLogCategory(#ClassName)
#else // 无操作
# define DECLARE_STATIC_LOG()
# define DEFINE_STATIC_LOG(ClassName)
#endif void suck(const char * fmt, ...); // 各日志子类宏定义
#if defined(_LOG4CPP) && defined(WIN32)
# define MakePrefix std::string(__FUNCTION__).append("() - ")
# define LogDebug (LogDebuger(log, std::string("[DEBUG]").append(MakePrefix)))
# define LogInfo (LogInfoer(log, std::string("[INFO]").append(MakePrefix)))
# define LogNotice (LogNoticer(log, std::string("[NOTICE]").append(MakePrefix)))
# define LogError (LogErrorer(log, std::string("[ERROR]").append(MakePrefix)))
#elif defined(_LOG4CPP) && !defined(WIN32)
# define MakePrefix(fmt) std::string(__FILE__).append(":s:").append(__FUNCTION__).append("() - ").append(fmt).c_str()
# define LogDebug(fmt, ...) log.debug(MakePrefix(fmt), ##__VA_ARGS__)
# define LogInfo(fmt, ...) log.info(MakePrefix(fmt), ##__VA_ARGS__)
# define LogNotice(fmt, ...) log.notice(MakePrefix(fmt), ##__VA_ARGS__)
# define LogError(fmt, ...) log.error(MakePrefix(fmt), ##__VA_ARGS__)
#else
# define LogDebug suck
# define LogInfo suck
# define LogNotice suck
# define LogError suck
#endif } #endif

LogUtils.cpp

#include "LogUtils.h"

#if defined(_LOG4CPP)

#include <log4cpp/PropertyConfigurator.hh>

// appenders

#include <log4cpp/Appender.hh>
#include <log4cpp/OstreamAppender.hh>
#include <log4cpp/FileAppender.hh>
#include <log4cpp/RollingFileAppender.hh>
#include <log4cpp/AbortAppender.hh>
#ifdef WIN32
#include <log4cpp/Win32DebugAppender.hh>
#include <log4cpp/NTEventLogAppender.hh>
#endif
#include <log4cpp/RemoteSyslogAppender.hh>
#ifdef LOG4CPP_HAVE_LIBIDSA
#include <log4cpp/IdsaAppender.hh>
#endif // LOG4CPP_HAVE_LIBIDSA #ifdef LOG4CPP_HAVE_SYSLOG
#include <log4cpp/SyslogAppender.hh>
#endif // layouts #include <log4cpp/Layout.hh>
#include <log4cpp/BasicLayout.hh>
#include <log4cpp/SimpleLayout.hh>
#include <log4cpp/PatternLayout.hh> #include <log4cpp/Priority.hh> #endif namespace dk
{
// 从指定的配置文件logfile中读取配置信息,并初始化log4cpp
void InitializeLog4cpp(const std::string & logfile)
{ #if defined(_LOG4CPP)
try
{
log4cpp::PropertyConfigurator::configure(logfile);
}
catch (log4cpp::ConfigureFailure & f)
{
std::cerr << "Configure Problem " << f.what() << std::endl; //#if defined(WIN32) // log4cpp::Appender * appender = new log4cpp::Win32DebugAppender("console"); //#else log4cpp::Appender * appender = new log4cpp::OstreamAppender("console", &std::cout);
//#endif log4cpp::PatternLayout * patternLayout = new log4cpp::PatternLayout();
patternLayout->setConversionPattern("%d [%t] %p - %m%n"); appender->setLayout(patternLayout); log4cpp::Category & root = log4cpp::Category::getRoot();
root.addAppender(appender);
root.setPriority(log4cpp::Priority::DEBUG);
}
#endif
} void ReleaseLog4cpp()
{
#if defined(_LOG4CPP)
log4cpp::Category::shutdown();
#endif
} #if defined(_LOG4CPP)
log4cpp::Category & GetLogCategory(const char * categoryName)
{
std::string name = "Logger.";
name.append(categoryName); // 这个name与配置文件相关,如果配置文件中包含了同名的category定义,则使用对应配置;否则将新建一个NOTSET类型的category,并将日志内容存入到Logger指定的输出端里。(因此,配置文件里必须有名为 Logger的category配置)
log4cpp::Category & category = log4cpp::Category::getInstance(name);
return category;
}
#endif void suck(const char * fmt, ...)
{
if (fmt) {}
} }

LogTracer.h

#ifndef _LOG_TRACER_H_
#define _LOG_TRACER_H_ // 开启 日志记录 宏定义
#define _LOG4CPP #if defined(_LOG4CPP) && defined(WIN32) #include <log4cpp/Category.hh> namespace log4cpp
{
class Category;
}; namespace dk
{ #include <stdarg.h>
#include <stdio.h> class LogTracer
{
public:
LogTracer(log4cpp::Category & log, const std::string & prefix)
: mLog(log), mMsg(prefix)
{} void operator()(const char * fmt, ...)
{
va_list ap;
va_start(ap, fmt);
AppendString(mMsg, fmt, ap);
WriteLog(mMsg);
va_end(ap);
} private:
virtual void WriteLog(const std::string & message) = ; void AppendString(std::string & message, const char * format, va_list args)
{
size_t size = ;
char * buffer = new char[size]; while ()
{
int n = _vsnprintf(buffer, size, format, args);
// If that worked, return a string.
if ((n > -) && (static_cast<size_t>(n) < size))
{
message.append(buffer);
delete [] buffer;
return;
} // Else try again with more space.
size = (n > -) ? n + : // ISO/IEC 9899:1999
size * ; // twice the old size
delete [] buffer;
buffer = new char[size];
}
} private:
// copy-constructor and operator=
LogTracer(const LogTracer &);
LogTracer & operator=(const LogTracer &); protected:
log4cpp::Category & mLog;
std::string mMsg;
}; class LogDebuger : public LogTracer
{
public:
LogDebuger(log4cpp::Category & log, const std::string & prefix)
: LogTracer(log, prefix)
{
} private:
virtual void WriteLog(const std::string & message)
{
mLog.debug(message);
}
}; class LogInfoer : public LogTracer
{
public:
LogInfoer(log4cpp::Category & log, const std::string & prefix)
: LogTracer(log, prefix)
{
} private:
virtual void WriteLog(const std::string & message)
{
mLog.info(message);
}
}; class LogNoticer : public LogTracer
{
public:
LogNoticer(log4cpp::Category & log, const std::string & prefix)
: LogTracer(log, prefix)
{
} private:
virtual void WriteLog(const std::string & message)
{
mLog.notice(message);
}
}; class LogErrorer : public LogTracer
{
public:
LogErrorer(log4cpp::Category & log, const std::string & prefix)
: LogTracer(log, prefix)
{
} private:
virtual void WriteLog(const std::string & message)
{
mLog.error(message);
}
}; } #endif
#endif

log4cpp.Logger.property  配置文件示例

# property configurator test file

log4cpp.rootCategory=DEBUG, rootAppender
log4cpp.category.Logger= DEBUG, A1 log4cpp.appender.rootAppender=ConsoleAppender
log4cpp.appender.rootAppender.layout=BasicLayout log4cpp.appender.A1=FileAppender
log4cpp.appender.A1.fileName=Log\\Visualization.log
log4cpp.appender.A1.layout=PatternLayout
log4cpp.appender.A1.layout.ConversionPattern=%d %m %n

调用示例:

#include "LogUtils.h"
#include "Demo.h" int main(int argc, char *argv[])
{ dk::InitializeLog4cpp("Log\\log4cpp.Logger.property"); {
Demo d;
d.TestIt(, );
} dk::ReleaseLog4cpp(); }

【转】Log4cpp 封装的更多相关文章

  1. log4cpp单例类封装

    body, table{font-family: 微软雅黑; font-size: 13.5pt} table{border-collapse: collapse; border: solid gra ...

  2. 对log4cpp进一步封装

    //Mylog.h如下: #pragma once//#include <iostream>#include <log4cpp/Category.hh>#include < ...

  3. 快速使用Log4Cpp

    封了一下接口,快速使用. 其他的你都不用管了. 这里封装了需要读取外部conf文件配置输出项.否则可以用getInstance初始化日志类 #include "L4Cpp.h" v ...

  4. log4cpp退出时内存泄露的修复方案

    1.缘由 一直对log4cpp非常有好感,就在自己的项目中集成了log4cpp1.1.1版本,并围绕着它建立了一系列的封装函数方便外部调用.写完了一个测试代码后,忽然想看看自己写的程序有没有内存泄露问 ...

  5. C++开源库(一) ----log4cpp详解

    我们在写程序的时候通常会希望将一些信息记录下来,方便我们进行日后的一些信息跟踪,错误排查等等.比如:我们在进行数据库操作的时候,我们通常希望知道现在是程序的哪一部分进行了数据库的操作,所以我们会记录下 ...

  6. log4cpp安装使用

    1. 主页:http://log4cpp.sourceforge.net“Log4cpp is library of C++ classes for flexible logging to files ...

  7. Windows下MinGW跨平台编译和使用log4cpp

    Log4cpp 是C++开源日志库,为 C++ 应用程序开发中提供了日志的追踪和调试功能,基于 LGPL 开源协议,移植自 java 的日志项目 log4j, 并在 api 上保持了一致性. 1. 环 ...

  8. C++ log4cpp使用(转)

    参考文章: 1.常用C++库(1)日志库 https://blog.csdn.net/qilimi1053620912/article/details/87378707 2.一步步入门log4cpp  ...

  9. [C#] 简单的 Helper 封装 -- RegularExpressionHelper

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

随机推荐

  1. jQuery实现多级手风琴树形下拉菜单(源码)

    前几天因为公司的菜单要调整,公司的UI框架是不支持的,所以就自己在网上找了一个下拉菜单,可以支持多级菜单数据的,菜单数据是从xml文件中配置后读取的,网上有许多这方面的例子感觉不是很好用,就打了个包贴 ...

  2. dense_rank()+hash提示改写优化SQL

    数据库环境:SQL SERVER 2005 今天看到一条SQL,返回10条数据,执行了50多S.刚好有空,就对它进行了优化,优化后1S出结果. 先看下原始SQL SELECT t1.line_no , ...

  3. jQuery 如何设置input checkbox 更有效 prop()

    问题:经常使用jQuery插件的attr方法获取checked属性值,获取的值的大小为未定义,此时可以用prop方法获取其真实值,下面介绍这两种方法的区别: 1.通过prop方法获取checked属性 ...

  4. CI 笔记,借鉴的4个辅助自定义函数

    在System的core的common.php中,借鉴的4个自定义函数, 摘自后盾网的CI教程 /** * 格式化打印函数 * @param [type] $arr [数组] * @return [t ...

  5. js作用域链

    js作用域链 <script> var up = 555; function display(){ var innerVar = 2; function inner(){ var inne ...

  6. C++ trivial和non-trivial构造函数及POD类型(转)

    原博客地址http://blog.csdn.net/a627088424/article/details/48595525 最近正纠结这个问题就转过来了,做了点补充(参考<深度探索C++对象模型 ...

  7. phpExcel使用与中文处理教程

    PHPExcel 是相当强大的 MS Office Excel 文档生成类库,当需要输出比较复杂格式数据的时候,PHPExcel 是个不错的选择.不过其使用方法相对来说也就有些繁琐. phpExcel ...

  8. 【JQuery学习历程】1.初识JQuery

    1.JQuery简介: JQuery是用js写的JavaScript库,是为了简化js对HTML元素的操作.实现动画效果并方便为网站提供ajax交互: 2.ready()方法: ready()方法和j ...

  9. mssql 获取表结构信息

    SELECT (case when a.colorder=1 then d.name else null end) 表名, a.colorder 字段序号,a.name 字段名, (case when ...

  10. VC皮肤库之duilib

    首先是个国产的开源 的,directui 界面库,开放,共享,惠众,共赢,遵循bsd协议,可以免费用于商业项目,目前支持Windows 32 .Window CE.Mobile等平台. Duilib ...