一、简介

log4cplus是一款优秀的基于C/C++的开源日志库。log4cplus具有线程安全,不用但心在多线程状态下写日志问题;使用灵活,可通过配置文件设置日志级别下输出位置,还可以在程序运行时动态设置日志输出级别,随心所欲掌控日志的输出;以及多粒度控制的特点。通过将信息划分优先级使其可以面向程序调试、运行、测试、和维护等全生命周期; 可以选择将信息输出到屏幕、文件、NT event log、甚至是远程服务器;通过指定策略对日志进行定期备份。可以满足大部分开发者对日志系统需求,功能全面。

二、安装

下载地址:http://sourceforge.net/projects/log4cplus/files/log4cplus-stable/

tar -xjf log4cplus-1.0.4.tar.bz2(解压缩),切换到解压缩路径下
安装:./configure;make;make install
这里已经安装成功,默认的路径lib库路径是/usr/local/lib/,头文件的位置:/usr/local/include/log4cplus
将/usr/local/lib下的和log4cplus相关的库都拷贝到/usr/lib,将头文件加到/etc/profile下的CPLUS_INCLUDE_PATH

三、使用

虽然功能强大,应该说log4cplus用起来还是比较复杂的,为了更好地使用它,先介绍一下它的基本要素。

Layouts      :布局器,控制输出消息的格式.
Appenders    :挂接器,与布局器紧密配合,将特定格式的消息输出到所挂接的设备终端
               (如屏幕,文件等等)。
Logger       :记录器,保存并跟踪对象日志信息变更的实体,当你需要对一个对象进行
               记录时,就需要生成一个logger。
Categories   :分类器,层次化(hierarchy)的结构,用于对被记录信息的分类,层次中
               每一个节点维护一个logger的所有信息。
Priorities   :优先权,包括TRACE, DEBUG, INFO, WARNING, ERROR, FATAL。

示例1:

#include <log4cplus/logger.h>
#include <log4cplus/configurator.h>
#include <log4cplus/loggingmacros.h> #include <iostream> using namespace std;
int main(int argc, char **argv)
{
using namespace log4cplus;
BasicConfigurator config;
config.configure(); Logger logger = Logger::getInstance("main");
LOG4CPLUS_ERROR(logger, "Hello, error!");
LOG4CPLUS_WARN(logger, "Hello, warn!");
LOG4CPLUS_INFO(logger, "Hello, info!");
LOG4CPLUS_DEBUG(logger, "Hello, debug!"); return ;
}

这个例子是最简单的,只使用到了Logger记录器。默认输出到屏幕。输出格式如下:

示例二:

#include <log4cplus/logger.h>
#include <log4cplus/consoleappender.h>
#include <log4cplus/loggingmacros.h> #include <iostream> using namespace std;
int main(int argc, char **argv)
{
using namespace log4cplus; Logger logger = Logger::getInstance("main");
SharedAppenderPtr appender(new ConsoleAppender());
std::string pattern = "%D{%m/%d/%y %H:%M:%S} %p - %m%n";
std::auto_ptr<Layout> layout(new PatternLayout(pattern));
appender->setLayout(layout);
logger.addAppender(appender); LOG4CPLUS_ERROR(logger, "Hello, error!");
LOG4CPLUS_WARN(logger, "Hello, warn!");
LOG4CPLUS_INFO(logger, "Hello, info!");
LOG4CPLUS_DEBUG(logger, "Hello, debug!"); return ;
}

这个例子做了改进,我们使用到了Appender,Layout。上面说过,Layout用来控制日志的输出格式,有三种Layout:SimpleLayout、PatternLayout、TTCCLayout。Appender用来控制日志输出的地方,比如:控制台、文件等。而且Layout是依附于Appender的,所以,如果我们想要自己控制日志格式,那么必须先获取一个Appender,然后吧

/**
* Set the layout for this appender. Note that some appenders have
* their own (fixed) layouts or do not use one. For example, the
* SocketAppender ignores the layout set here.
*/
virtual void setLayout(std::auto_ptr<Layout> layout);

这个Appender类的setLayout函数,也就是设置Layout的函数。它需要的参数是一个智能指针。

virtual void addAppender(SharedAppenderPtr newAppender);

这个是Logger类中设置Appender的函数,它需要的参数是一个SharedAppenderPtr类型。

所以如果我们要使用Appender和Layout,就需要构造这样的类型,并传入到对应的函数中去。

因为Layout是需要依附于Appender的,所以,这里我们使用了一个ConsoleAppender类以及PatternLayout类。

输出格式如下:

相比于示例1,这里的每一条日志前面都输出了时间。这就是使用Layout的效果,这里先详细介绍一个PatternLayout的使用方法,之后再介绍另外两种Layout。

PatternLayout:

可以看出通过填写特定格式的模式字符串"pattern",原始信息被包含到一堆有格式的信息当中了,这就使得
用户可以根据自身需要来定制显示内容。"pattern"可以包含普通字符串和预定义的标识符,其中:

(1)普通字符串,能够被直接显示的信息。
(2)预定义标识符,通过"%"与一个或多个字符共同构成预定义的标识符,能够产生出特定格式信息。

关于预定义标识符,log4cplus文档中提供了详细的格式说明,我每种都试了一下,以上述代码为例,根据不同
的pattern,各种消息格式使用情况列举如下:

(1)"%%",转义为%, 即,std::string pattern = "%%" 时输出: "%"
(2)"%c",输出logger名称,比如std::string pattern ="%c" 时输出: "test_logger.subtest",
     也可以控制logger名称的显示层次,比如"%c{1}"时输出"test_logger",其中数字表示层次。
(3)"%D",显示本地时间,当std::string pattern ="%D" 时输出:"2004-10-16 18:55:45",%d显示标准时间,
     所以当std::string pattern ="%d" 时输出 "2004-10-16 10:55:45" (因为我们是东8区,差8个小时啊)。
     可以通过%d{...}定义更详细的显示格式,比如%d{%H:%M:%s}表示要显示小时:分钟:秒。大括号中可显示的
     预定义标识符如下:

%a -- 表示礼拜几,英文缩写形式,比如"Fri"
%A -- 表示礼拜几,比如"Friday"
%b -- 表示几月份,英文缩写形式,比如"Oct"
%B -- 表示几月份,"October"
%c -- 标准的日期+时间格式,如 "Sat Oct 16 18:56:19 2004"
%d -- 表示今天是这个月的几号(-)""
%H -- 表示当前时刻是几时(-),如 ""
%I -- 表示当前时刻是几时(-),如 ""
%j -- 表示今天是哪一天(-),如 ""
%m -- 表示本月是哪一月(-),如 ""
%M -- 表示当前时刻是哪一分钟(-),如 ""
%p -- 表示现在是上午还是下午, AM or PM
%q -- 表示当前时刻中毫秒部分(-),如 ""
%Q -- 表示当前时刻中带小数的毫秒部分(-999.999),如 "430.732"
%S -- 表示当前时刻的多少秒(-),如 ""
%U -- 表示本周是今年的第几个礼拜,以周日为第一天开始计算(-),如 ""
%w -- 表示礼拜几,(-, 礼拜天为0),如 ""
%W -- 表示本周是今年的第几个礼拜,以周一为第一天开始计算(-),如 ""
%x -- 标准的日期格式,如 "10/16/04"
%X -- 标准的时间格式,如 "19:02:34"
%y -- 两位数的年份(-),如 ""
%Y -- 四位数的年份,如 ""
%Z -- 时区名,比如 "GMT"

(4)"%F",输出当前记录器所在的文件名称,比如std::string pattern ="%F" 时输出: "main.cpp"
(5)"%L",输出当前记录器所在的文件行号,比如std::string pattern ="%L" 时输出: "51"
(6)"%l",输出当前记录器所在的文件名称和行号,比如std::string pattern ="%L" 时输出:
     "main.cpp:51"
(7)"%m",输出原始信息,比如std::string pattern ="%m" 时输出: "teststr",即上述代码中
     LOG4CPLUS_DEBUG的第二个参数,这种实现机制可以确保原始信息被嵌入到带格式的信息中。
(8)"%n",换行符,没什么好解释的
(9)"%p",输出LogLevel,比如std::string pattern ="%p" 时输出: "DEBUG"
(10)"%t",输出记录器所在的线程ID,比如std::string pattern ="%t" 时输出: "1075298944"
(11)"%x",嵌套诊断上下文NDC (nested diagnostic context) 输出,从堆栈中弹出上下文信息,NDC可以用对
      不同源的log信息(同时地)交叉输出进行区分,关于NDC方面的详细介绍会在下文中提到。
(12)格式对齐,比如std::string pattern ="%-10m"时表示左对齐,宽度是10,此时会输出"teststr   ",当
      然其它的控制字符也可以相同的方式来使用,比如"%-12d","%-5p"等等(刚接触log4cplus文档时还以为
      "%-5p"整个字符串代表LogLevel呢,呵呵)。

SimpleLayout:

是一种简单格式的布局器,在输出的原始信息之前加上LogLevel和一个"-"。就是示例1的输出形式。

TTCCLayout:

是在PatternLayout基础上发展的一种缺省的带格式输出的布局器, 其格式由时间,线程ID,Logger和NDC 组
成(consists of time, thread, Logger and nested diagnostic context information, hence the name),
因而得名(怎么得名的?Logger里哪里有那个"C"的缩写啊!名字起得真够烂的,想扁人)。提供给那些想显示
典型的信息(一般情况下够用了)又懒得配置pattern的同志们。

示例三:

下面我们开始尝试将日志记录到文件中,这里需要用到文件相关的Appender类,log4cplus提供了三个这样的类:FileAppender类、RollingFileAppender类、DailyRollingFileAppender类。

FileAppender类:

实现了基本的文件操作,构造函数如下:

FileAppender(const log4cplus::tstring& filename,
std::ios_base::openmode mode = std::ios_base::trunc,
bool immediateFlush = true);
filename : 文件名
mode : 文件类型,可选择的文件类型包括app、ate、binary、in、out、trunc,因为实际上只是对stl的一个简单包装,呵呵,这里就不多讲了。缺省是trunc,表示将先前文件删除。
immediateFlush :缓冲刷新标志,如果为true表示每向文件写一条记录就刷新一次缓存,否则直到FileAppender被关闭或文件缓存已满才更新文件,一般是要设置true的,比如你往文件写的过程中出现了错误(如程序非正常退出),即使文件没有正常关闭也可以保证程序终止时刻之前的所有记录都会被正常保存。

RollingFileAppender类:

RollingFileAppender类可以根据你预先设定的大小来决定是否转储,当超过该大小,后续log信息会另存到新
文件中,除了定义每个记录文件的大小之外,你还要确定在RollingFileAppender类对象构造时最多需要多少个
这样的记录文件(maxBackupIndex+1),当存储的文件数目超过maxBackupIndex+1时,会删除最早生成的文件,
保证整个文件数目等于maxBackupIndex+1。然后继续记录

构造函数如下:

RollingFileAppender(const log4cplus::tstring& filename,
long maxFileSize = **, // 10 MB
int maxBackupIndex = ,
bool immediateFlush = true);
filename : 文件名
maxFileSize : 文件的最大尺寸
maxBackupIndex : 最大记录文件数
immediateFlush : 缓冲刷新标志

log4cplus中隐含定义了文件的最小尺寸是200K,只有大于200K的设置才生效,<= 200k的设置都会被认为是
200K.

DailyRollingFileAppender类:

DailyRollingFileAppender类可以根据你预先设定的频度来决定是否转储,当超过该频度,后续log信息会另存
到新文件中,这里的频度包括:MONTHLY(每月)、WEEKLY(每周)、DAILY(每日)、TWICE_DAILY(每两天)、
HOURLY(每时)、MINUTELY(每分)

构造函数如下:

DailyRollingFileAppender(const log4cplus::tstring& filename,
DailyRollingFileSchedule schedule = DAILY,
bool immediateFlush = true,
int maxBackupIndex = );
filename : 文件名
schedule : 存储频度
immediateFlush : 缓冲刷新标志
maxBackupIndex : 最大记录文件数

这里我们就使用FileAppender来将日志写入到文件中,代码如下:

#include <log4cplus/logger.h>
#include <log4cplus/fileappender.h>
#include <log4cplus/loggingmacros.h> #include <iostream> using namespace std;
int main(int argc, char **argv)
{
using namespace log4cplus; Logger logger = Logger::getInstance("main");
SharedAppenderPtr appender(new FileAppender("example3.log"));
std::string pattern = "%D{%m/%d/%y %H:%M:%S} %p - %m%n";
std::auto_ptr<Layout> layout(new PatternLayout(pattern));
appender->setLayout(layout);
logger.addAppender(appender); LOG4CPLUS_ERROR(logger, "Hello, error!");
LOG4CPLUS_WARN(logger, "Hello, warn!");
LOG4CPLUS_INFO(logger, "Hello, info!");
LOG4CPLUS_DEBUG(logger, "Hello, debug!"); return ;
}

相比于示例2,只是修改了一下Appender的类型。

输出结果:

四、优先级控制

日志系统的另一个基本功能就是能够让使用者能够按照自己的意愿控制什么时候,哪些信息可以输出。如果能够让用户在任意时刻设置允许输出的loglevel信息就好了。log4cplus通过LoglevelManager、LogLog、Filter三种方式实现了上述功能。

在研究LogLevelManager之前,首先介绍一下log4cplus中logger的存储机制,在log4cplus中,所有logger都通过一个层次化的结构(其实内部是hash表)来组织的,有一个Root级别的logger,可以通过一下方法获取。

Logger root = Logger::getRoot();

用户定义的logger都有一个名字与之对应,比如:

Logger test = Logger::getInstance("test");

可以定义该logger的子logger:

Logger subTest = Logger::getInstance("test.subtest");

注意:Root级别的logger只有通过getRoot()方法获取,Logger::getInstance("root")获得的是它的子对象而已。有了这些具有父子关系的logger之后可以分别设置其LogLevel,比如:

root.setLogLevel(...);
Test.setLogLevel(...);
subTest.setLogLevel(...);

LogLevelManager负责设置logger的优先级,各个logger可以通过setLogLevel设置自己的优先级,
当某个logger的LogLevel设置成NOT_SET_LOG_LEVEL时,该logger会继承父logger的优先级,另外,
如果定义了重名的多个logger, 对其中任何一个的修改都会同时改变其它logger。

参考:

http://honey-bee.iteye.com/blog/65805

http://blog.csdn.net/hmalloc/article/details/8443291

CentOS 6.2下log4cplus的使用的更多相关文章

  1. 不要着急改代码,先想想--centos 6.8下编译安装tmux

    诸位读者新年好,2017开年第一篇博客,请允许我先问候一下看到这篇博客的诸位.写博客是我2017年定下的目标之一,希望我会坚持下去. 最近打算尝试一下tmux这个神器,于是有了这一篇关于思维方式的Bl ...

  2. CentOS 6.3下 安装 Mono 3.2 和Jexus 5.4

    最新更新参看: Centos 7.0 安装Mono 3.4 和 Jexus 5.6 2012年初写过一篇<32和64位的CentOS 6.0下 安装 Mono 2.10.8 和Jexus 5.0 ...

  3. CentOS 6.3下Samba服务器的安装与配置方法(图文详解)

    这篇文章主要介绍了CentOS 6.3下Samba服务器的安装与配置方法(图文详解),需要的朋友可以参考下   一.简介  Samba是一个能让Linux系统应用Microsoft网络通讯协议的软件, ...

  4. CentOS 6.4下编译安装MySQL 5.6.14

    概述: CentOS 6.4下通过yum安装的MySQL是5.1版的,比较老,所以就想通过源代码安装高版本的5.6.14. 正文: 一:卸载旧版本 使用下面的命令检查是否安装有MySQL Server ...

  5. CentOS个人目录下中文路径转英文路径

    CentOS个人目录下中文路径转英文路径 如果安装了中文版到CentOS之后,root目录及home目录下会出现中文到路径名,如"桌面"."文档"," ...

  6. CentOS 6.3下配置LVM(逻辑卷管理)

    一.简介 LVM是逻辑盘卷管理(Logical Volume Manager)的简称,它是Linux环境下对磁盘分区进行管理的一种机制,LVM是建立在硬盘和分区之上的一个逻辑层,来提高磁盘分区管理的灵 ...

  7. CentOS 6.3下配置软RAID(Software RAID)

    一.RAID 简介 RAID 是英文Redundant Array of Independent Disks 的缩写,翻译成中文意思是“独立磁盘冗余阵列”,有时也简称磁盘阵列(Disk Array). ...

  8. CentOS 6.4 下安装vsftpd

    概述: vsftpd是Linux下比较著名的FTP服务器,搭建FTP服务器当然首选这个. 本文介绍了在CentOS 6.4下安装vsftpd.配置虚拟用户登录FTP的过程. 正文: 一:安装vsftp ...

  9. centos 6.5下安装mysql+nginx+redmine 3.1.0 笔记

    centos 6.5下安装mysql+nginx+redmine 3.1.0 笔记 目录[-] 过程 1.安装RVM 2.利用rvm安装 Ruby 1.9.3 并设为默认 3.安装rails 4.安装 ...

随机推荐

  1. Spark RDD API扩展开发

    原文链接: Spark RDD API扩展开发(1) Spark RDD API扩展开发(2):自定义RDD 我们都知道,Apache Spark内置了很多操作数据的API.但是很多时候,当我们在现实 ...

  2. Host 'localhost' has multiple addresses. 解决办法

    phpstorm调试php 错误提示: Host 'localhost' has multiple addresses. You must choose one explicitly!Couldn't ...

  3. 使用YUI+Ant 实现JS CSS压缩

    今天研究了一下YUI yahoo开源框架,感觉很猛啊. 于是乎我做了一个YUI的ant实现,网上好多关于bat的实现,我就另辟蹊径,出个关于这个的ant实现,嘿嘿独一无二的文章,如果转载的话,其注明作 ...

  4. Javascript里面的时间处理:将时间戳或时间对象转成字符串格式

    问题背景:想把一个时间直接转成字符串格式 通过查api发现有个toLocaleString(),根据本地时间格式,把 Date 对象转换为字符串 new Date().toLocaleString() ...

  5. IOS NSNotification Center 通知中心的使用

    通知中心,它是IOS程序内部的一种消息广播机制,通过它,可以实现无引用关系的对象之间的通信.通知中心他是基于观察者模式,它只能进行程序内部通信,不能跨应用程序进程通信.当通知中心接受到消息后会根据设置 ...

  6. Android -- SurfaceView绘制

    SurfaceView SurfaceView是View的一个特殊子类,它的目的是另外提供一个线程进行绘制操作. 步骤 1.用SurfaceView进行绘制,首先要创建一个类,继承 SurfaceVi ...

  7. 开源力量:微软竟开源 PowerShell

    导读 曾经有段时间,微软称 Linux 是“癌症”,但是随着时光流逝,现在微软已经认识到了开源世界的不断增长,除了在这个领域加大投入之外别无选择.微软已经启动了几个开源项目,希望能吸引一些 Linux ...

  8. Woocommerce 分类下的产品如何使用ID号来作为默认排序字段

    在给一个客户开发网店系统时使用了WordPress系统的Woocommerce插件 WordPress版本:3.8 Woocommerce版本:2.0.20 如果没有指定排序规则(指定的字段),则Wo ...

  9. angularjs中ajax请求时传递参数的方法

    method1方法使用的是params参数,该用法会把参数直接附加到url中 method2方法使用的是data参数,该参数会把页面参数类型从默认的multipart/form-data改为appli ...

  10. 网络电台-SHOUTcast

    网络电台种类 目前的网络电台网站一般是基于以下三种协议的: mms.rtsp.http 其中mms是微软公司提出的网络流媒体协议,通常采用wma格式的文件,Android现在还不支持这种协议,也不支持 ...