引用于:http://blog.csdn.net/focusonace/article/details/3108873

    http://peirenlei.iteye.com/blog/305036

介绍了ACE中日志系统(Logging Facility)的基本使用

一、简介

ACE有自己的Logging Facility,与log4cplus相似,ACE日志系统具有线程安全、灵活、分级显示等特点,可以面向程序调试、运行、测试、和维护等全生命周期,可以选择将信息输出到屏幕、文件、系统日志(如Windows下的Event log)、甚至是远程服务器。除此之外,ACE日志系统支持回调函数以及运行时刻动态配置(Runtime Configuration ),本文主要参考了《ACE Programmer's Guide,The: Practical Design Patterns for Network and Systems Programming》。

二、ACE日志系统的基本使用

首先通过一个引例介绍如何调用ACE Log Macro:
#include < ace/Log_Msg.h >
int ACE_TMAIN (int, ACE_TCHAR *[])
{
  ACE_DEBUG ((LM_INFO, ACE_TEXT ("Hi ACE Logging Facility!\n")));
  return 0;
}
程序需要包含头文件:Log_Msg.h,该文件定义了各种有用的日志输出宏,宏ACE_TMAIN 是ACE定义的
wide-character-enabled entry point ,ACE日志系统缺省输出到stderr,所以对于Console类型的应用程序,
会直接输出到屏幕。运行该程序时屏幕上会显示:
Hi ACE Logging Facility!
可以看出ACE日志系统的宏调用非常简单:
ACE_XXX((severity, formatting-args));
其中,ACE_XXX表示日志输出宏(ACE Logging Macros),包括:
ACE_ERROR((level, string, ...))
ACE_DEBUG((level, string, ...))
ACE_TRACE(string)
ACE_ASSERT(test)
ACE_HEX_DUMP((level, buffer, size [,text]))
ACE_RETURN(value)
ACE_ERROR_RETURN((level, string, ...), value)
ACE_ERROR_INIT( value, flags )
ACE_ERROR_BREAK((level, string, ...))
severity即输出的严重等级(Severity Level),包括:
 LM_SHUTDOWN = 01,系统死机级别 
 LM_TRACE = 02,跟踪级别 
 LM_DEBUG = 04,DEBUG级别 
 LM_INFO = 010,常规信息级别 
 LM_NOTICE = 020,注意级别 
 LM_WARNING = 040,警告级别 
 LM_STARTUP = 0100,启动级别 
 LM_ERROR = 0200,错误级别 
 LM_CRITICAL = 0400,危急级别 
 LM_ALERT = 01000,可修复的警告级别 
 LM_EMERGENCY = 02000,全局警告级别 
formatting-args是要输出的内容,格式类似于printf函数的输出格式:
Code        Argument Type   Displays
A           ACE_timer_t     浮点数
a           —              导致程序终止(Abort)
c           char            单个字符
C           char*           字符串(narrow characters)
i,d         int             10进制整数
I           —              缩进
e,E,f,F,g,G double          双精度浮点数
l           —              行号
M           —              severity level的名称
m           —              错误号(errorno)
N           —              文件名
n           —              ACE_Log_Msg::open()指定的程序名
o           int             八进制整数
P           —              当前进程ID
p           ACE_TCHAR*      字符串,后接错误号,同perror
Q           ACE_UINT64      64位无符号十进制整数
r           void (*)()      函数调用
R           int             十进制整数
S           int             数字对应的信号名称
s           ACE_TCHAR*      ACE_TCHAR类型的字符串
T           —              当前时间(hour:minute:sec.usec)
D           —              时戳(month/day/year/hour:minute:sec.usec)
t           —              当前线程ID
u           int             无符号十进制整数
w           wchar_t         Single wide character
W           wchar_t*        Wide-character string
x,X         int             十六进制数
@           void*           指针(十六进制)
%           N/A             %
我们可以通过一些例子来进一步了解如何使用这些宏、Severity Level和格式,在开始之前,首先澄清一个概念:
一些日志系统(如log4cplus)是靠宏名来区分日志信息类型的,比如:
    LOG4CPLUS_DEBUG(Logger::getRoot(),"some text")   /* DEDUG 类型 */
    LOG4CPLUS_ERROR(Logger::getRoot(),"some text")   /* ERROR 类型 */
    
ACE的日志系统不是靠宏名本身,而是靠宏的第一个参数,即Severity Level来区分日志信息类型:
    ACE_DEBUG((LM_DEBUG, ACE_TEXT ("some text\n"))); /* DEDUG 类型 */
    ACE_ERROR((LM_DEBUG, ACE_TEXT ("some text\n"))); /* DEDUG 类型 */
    ACE_DEBUG((LM_ERROR, ACE_TEXT ("some text\n"))); /* ERROR 类型 */
    ACE_ERROR((LM_ERROR, ACE_TEXT ("some text\n"))); /* ERROR 类型 */
从这个角度考虑,宏ACE_DEBUG和ACE_ERROR没有什么区别,几乎可以混用,这两个宏的区别主要体现在两方面:
a) 语义上的差别
b) 语句执行完后,op_status不同,比如:
    ACE_Log_Msg *lm = ACE_LOG_MSG;
    ACE_DEBUG ((LM_ERROR, ACE_TEXT ("some text\n")));
    ACE_ASSERT( 0 == lm->op_status() );  /* ACE_DEBUG执行完后,op_status 是0 */
    ACE_ERROR ((LM_ERROR, ACE_TEXT ("some text\n")));
    ACE_ASSERT( -1 == lm->op_status() ); /* ACE_ERROR执行完后,op_status 是-1 */
    
下面举一些典型用例:
【use-case1】 如何在编译时刻Enabling 或Disabling Logging Macros
在#include < ace/Log_Msg.h >之前定义一些宏,可以在编译时刻Enable/Disable logging Macros
/* disable ACE_DEBUG and ACE_ERROR */
#define ACE_NLOGGING
/* enable ACE_DEBUG and ACE_ERROR(缺省) */
#undef ACE_NLOGGING
/* enable ACE_ASSERT(缺省) */
#undef ACE_NDEBUG
/* disable ACE_ASSERT */
#define ACE_NDEBUG
/* enable ACE_TRACE */
#define ACE_NTRACE 0
/* disable ACE_TRACE(缺省) */
#define ACE_NTRACE 1
【use-case2】格式使用
a) 打印完结果后终止(Abort)
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("begin %a end \n")));
输出以下信息后提示Abort信息:
begin Aborting... end
b) 缩进
int ACE_TMAIN (int, ACE_TCHAR *[])
{
    ACE_TRACE("main");
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("%Itext\n")));
    return 0;
}
输出:
(700) calling main in file `D:\aceprj\testlog\main.cpp' on line 24
    text
(700) leaving main
(其中,700是ACE_TRACE控制输出的当前线程ID)
c) HEX Dump(调试时候比较有用)
    char szBuf[128] = "hello world!";
    ACE_HEX_DUMP((LM_DEBUG, szBuf, 128,"szBuf: "));
输出:
szBuf:  - HEXDUMP 128 bytes
68 65 6c 6c 6f 20 77 6f  72 6c 64 21 00 00 00 00   hello world!....
00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00   ................
00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00   ................
00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00   ................
00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00   ................
00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00   ................
00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00   ................
00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00   ................
d)函数调用
void foo(void)
{
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("in func foo\n")));
}
int ACE_TMAIN (int, ACE_TCHAR *[])
{
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("call function %r\n"),foo));
    return 0;
}
输出:
in func foo
call function
e) 进制转换
    ACE_UINT64 uint64A = 210113198510308319;
    int hexA = 0xBEEE3F;
    int octA = 9;
    ACE_timer_t tmInterval = 0.000001;
    
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("UINT64: %Q\n"),uint64A));
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("HEX: %X, hex: %x\n"),hexA,hexA));
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("Ptr in hex:%@\n"),&hexA));
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("Octal number of 9:%o\n"),octA));
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("timer's Interval: %A\n"),tmInterval));
输出:
UINT64: 210113198510308319
HEX: BEEE3F, hex: beee3f
Ptr in hex:0012FE30
Octal number of 9:11
timer's Interval: 0.000001
f)综合使用
    ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("%D, %M, [%t], Line:%l\n")));
    ACE_DEBUG ((LM_ERROR, ACE_TEXT ("%D, %M, [%t], Line:%l\n")));
    
输出:
    Mon Mar 20 2005 20:02:16.912000, LM_DEBUG, [3296], Line:24
    Mon Mar 20 2006 20:02:16.912000, LM_ERROR, [3296], Line:25
    
    
【use-case3】宏使用
a) ACE_RETURN
int foo(void)
{
    ACE_RETURN(3);
}
int ACE_TMAIN (int, ACE_TCHAR *[])
{
    foo();
    ACE_ASSERT( 3 == ACE_LOG_MSG->op_status() );
    return 0;
}
输出:(无任何输出)
b) ACE_ERROR_RETURN
int foo(void)
{
    ACE_ERROR_RETURN((LM_DEBUG,"Return error\n"), 3);
}
int ACE_TMAIN (int, ACE_TCHAR *[])
{
    foo();
    ACE_ASSERT( 3 == ACE_LOG_MSG->op_status() );
    return 0;
}
输出:
Return error
c) ACE_ERROR_BREAK
    int i = 0;
    while(1)
    {
        if( i > 10 )
        {
            ACE_ERROR_BREAK((LM_DEBUG,"break from while\n"));
        }
        i++;
    }
    ACE_DEBUG((LM_DEBUG, "i = %d\n", i));
输出:
break from while
i = 11
【use-case4】 如何在运行时刻Enabling 或Disabling Logging Severities
由于缺省情况下在进程范围内所有logging severities都会被输出,因此需要通过priority_mask、
ACE_Log_Msg::enable_debug_messages、ACE_Log_Msg::disable_debug_messages来过滤输出。
其中priority_mask的用法如下:
    /* Get the current ACE_Log_Priority mask. */
    u_long priority_mask (MASK_TYPE = THREAD);
    /* Set the ACE_Log_Priority mask, returns original mask. */
    u_long priority_mask (u_long, MASK_TYPE = THREAD);
这里MASK_TYPE可选择:ACE_Log_Msg::PROCESS(作用范围是进程)或ACE_Log_Msg::THREAD(作用范围是线程)
enable_debug_messages/disable_debug_messages的用法如下:
    static void disable_debug_messages
    (ACE_Log_Priority priority = LM_DEBUG);
    static void enable_debug_messages
    (ACE_Log_Priority priority = LM_DEBUG);
这里输入参数可选择任意需要打开或屏蔽的logging severity,不像函数名称暗示的那样只能控制LM_DEBUG
类型,另外由于是静态函数,enable_debug_messages/disable_debug_messages的设置对整个进程有效。
注意一下priority_mask与enable_debug_messages/disable_debug_messages的区别:
priority_mask将进程(或线程)的logging severitys用其第一个参数所代替,清除之前的设置。
enable_debug_messages将进程的logging severitys用其参数去累加,不清除之前的设置。
disable_debug_messages将进程的logging severitys用其参数去屏蔽,不清除之前的设置。
a) 整个进程仅允许输出LM_ERROR和LM_WARNING
    ACE_DEBUG((LM_DEBUG, "Debug Text1\n"));
    ACE_DEBUG((LM_ERROR, "Error Text1\n"));
    
    ACE_LOG_MSG->priority_mask (LM_ERROR | LM_WARNING, ACE_Log_Msg::PROCESS);
    
    ACE_DEBUG((LM_DEBUG, "Debug Text2\n"));
    ACE_DEBUG((LM_ERROR, "Error Text2\n"));
    
或者:
    ACE_DEBUG((LM_DEBUG, "Debug Text1\n"));
    ACE_DEBUG((LM_ERROR, "Error Text1\n"));
    /* 禁用了进程所有的logging severitys */
    ACE_LOG_MSG->priority_mask (0, ACE_Log_Msg::PROCESS);
    ACE_Log_Msg::enable_debug_messages (LM_ERROR);
    ACE_Log_Msg::enable_debug_messages (LM_WARNING);
    ACE_DEBUG((LM_DEBUG, "Debug Text2\n"));
    ACE_DEBUG((LM_ERROR, "Error Text2\n"));
    
    
输出:
Debug Text1
Error Text1
Error Text2
b) 每个线程控制各自的logging severity
#include < ace/Task.h >    /* for ACE_Thread_Manager */
#include < ace/Log_Msg.h >
void service(void)
{
ACE_LOG_MSG->priority_mask (LM_INFO, ACE_Log_Msg::THREAD);
ACE_DEBUG((LM_DEBUG, "in service, MsgType:%M,  ThreadID: %t\n"));
ACE_DEBUG((LM_INFO,  "in service, MsgType:%M,  ThreadID: %t\n"));
}
void worker(void)
{
 ACE_LOG_MSG->priority_mask (LM_DEBUG, ACE_Log_Msg::THREAD);
ACE_DEBUG((LM_DEBUG, "in worker,  MsgType:%M, ThreadID: %t\n"));
ACE_DEBUG((LM_INFO,  "in worker,  MsgType:%M, ThreadID: %t\n"));
}
int ACE_TMAIN (int, ACE_TCHAR *[])
{
  ACE_LOG_MSG->priority_mask (0, ACE_Log_Msg::PROCESS);
 ACE_Thread_Manager::instance ()->spawn ((ACE_THR_FUNC)service);
ACE_Thread_Manager::instance ()->spawn_n (3, (ACE_THR_FUNC)worker);
ACE_Thread_Manager::instance ()->wait();
 return 0;
}
输出:
in service, MsgType:LM_INFO,  ThreadID: 3428
in worker,  MsgType:LM_DEBUG, ThreadID: 2064
in worker,  MsgType:LM_DEBUG, ThreadID: 1240
in worker,  MsgType:LM_DEBUG, ThreadID: 1800

ACE通过全局单体实例 ACE_LOG_MSG 管理和控制输出,我们可以利用这个实例把输出重定向

到文件,也可以重定向到标志输出 
其中还可以控制什么级别的消息可被输出,简单使用如下:

ACE_LOG_MSG->set_flags (ACE_Log_Msg::STDERR); 
  设置输出到标准错误输出 
  ACE_LOG_MSG->clr_flags (ACE_Log_Msg::STDERR); 
  关闭输出到标志错误输出 
  ACE_LOG_MSG->set_flags (ACE_Log_Msg::OSTREAM); 
  ofstream myostream (filename, ios::out | ios::trunc); 
  ACE_LOG_MSG->msg_ostream (&myostream); 
  设置输出到文件 
  u_long priority_mask =ACE_LOG_MSG->priority_mask (ACE_Log_Msg::PROCESS); 
  ACE_SET_BITS (priority_mask,LM_DEBUG |LM_INFO); 
  设置只记录LM_DEBUG 或者LM_INFO级别的消息 
  ACE_CLR_BITS (priority_mask,LM_DEBUG | LM_INFO);                    
  清除设置条件 
ACE_LOG_MSG还有一个非常有意思的功能,能以16进制方式输出内存中的块 ,简单使用如下: 
      ACE_LOG_MSG->log_hexdump (LM_DEBUG, 
                                (char *) array, 
                                sizeof array);

下面附上一个使用例子,例子来源于ACE包中的例子程序 
#include "ace/OS_main.h" 
#include "ace/streams.h" 
#include "ace/Log_Msg.h" 
int 
ACE_TMAIN (int, ACE_TCHAR *[]) 

  // 这个消息将输出到STDERR 
  ACE_DEBUG ((LM_DEBUG, 
              "first message/n")); 
  ACE_LOG_MSG->clr_flags (ACE_Log_Msg::STDERR); 
  // 这个消息将不被输出,因为标志位清除了 
  ACE_DEBUG ((LM_DEBUG, 
              "second message/n")); 
  ACE_LOG_MSG->set_flags (ACE_Log_Msg::OSTREAM); 
  //这个消息将不被输出,因为流的实体尚未定义 
  ACE_DEBUG ((LM_DEBUG, 
              "third message/n")); 
  //建立一个流实体,指向d:/output.log文件 
  const char *filename = "d://output.log"; 
  ofstream myostream (filename, ios::out | ios::trunc); 
  if (myostream.bad ()) 
    return 1; 
  // 设置输出到流 
  ACE_LOG_MSG->msg_ostream (&myostream); 
  //这个消息将输出到文件中 
  ACE_DEBUG ((LM_DEBUG, 
              "fourth message/n")); 
  ACE_LOG_MSG->set_flags (ACE_Log_Msg::STDERR); 
  // 这个消息将同时输出到文件和STDERR 
  ACE_DEBUG ((LM_DEBUG, 
              "fifth message/n")); 
  return 0; 
}

ACE日志系统的更多相关文章

  1. C++ 高性能无锁日志系统

    服务器编程中,日志系统需要满足几个条件 .高效,日志系统不应占用太多资源 .简洁,为了一个简单的日志功能引入大量第三方代码未必值得 .线程安全,服务器中各个线程都能同时写出日志 .轮替,服务器不出故障 ...

  2. Atitit.日志系统slf4j的使用

    Atitit.日志系统slf4j的使用 SLF4J: Class path contains multiple SLF4J bindings. SLF4J: Found binding in [jar ...

  3. Android源码——Logger日志系统

    Android的Logger日志系统是基于内核中的Logger日志驱动程序实现的. 日志保存在内核空间中 缓冲区保存日志   分类方法:日志的类型  +   日志的输出量   日志类型:   main ...

  4. 使用Slf4j集成Log4j2构建项目日志系统的完美解决方案

    一.背景 最近因为公司项目性能需要,我们考虑把以前基于的log4j的日志系统重构成基于Slf4j和log4j2的日志系统,因为,使用slf4j可以很好的保证我们的日志系统具有良好的兼容性,兼容当前常见 ...

  5. 日志系统实战(三)-分布式跟踪的Net实现

    介绍 在大型系统开发调试中,跨系统之间联调开始变得不好使了.莫名其妙一个错误爆出来了,日志虽然有记录,但到底是哪里出问题了呢? 是Ios端参数传的不对?还是A系统或B系统提供的接口导致?相信有不少人遇 ...

  6. 日志系统实战(一)—AOP静态注入

    背景 近期在写日志系统,需要在运行时在函数内注入日志记录,并附带函数信息,这时就想到用Aop注入的方式. AOP分动态注入和静态注入两种注入的方式. 动态注入方式 利用Remoting的Context ...

  7. Android性能优化之UncaughtExceptionHandler定制自己的错误日志系统

    前言: 每当我们app测试的时候,测试人员总是对我们说这里崩溃了,那里挂掉了!我们只能默默接受,然后尝试着重现bug,更可悲的是有时候bug很难复现,为了解决这种现状所以我们要尝试这建立一个自己的bu ...

  8. [Asp.net 5] Logging-其他日志系统的实现

    Microsoft.Framework.Logging.NLog 使用Nlog扩展日志系统:按照我们上节说的,对于扩展的日志系统都要实现俩个接口ILogger.ILoggerProvider.所以在当 ...

  9. 【转载】scribe、chukwa、kafka、flume日志系统对比

    原文地址:http://www.ttlsa.com/log-system/scribe-chukwa-kafka-flume-log-system-contrast/ 1. 背景介绍许多公司的平台每天 ...

随机推荐

  1. Python爬虫使用浏览器的cookies:browsercookie

    很多用Python的人可能都写过网络爬虫,自动化获取网络数据确实是一件令人愉悦的事情,而Python很好的帮助我们达到这种愉悦.然而,爬虫经常要碰到各种登录.验证的阻挠,让人灰心丧气(网站:天天碰到各 ...

  2. Python字符串所有操作函数

    name = "my \tname is {name} and i am {year} old" print(name.capitalize())#首字母大写 print(name ...

  3. Python 学习笔记之——用 sklearn 对数据进行预处理

    1. 标准化 标准化是为了让数据服从一个零均值和单位方差的标准正态分布.也即针对一个均值为 \(mean\) 标准差为 \(std\) 的向量 \(X\) 中的每个值 \(x\),有 \(x_{sca ...

  4. 通过Ajax上传文件至WebApi服务器

    https://stackoverflow.com/questions/43013858/ajax-post-a-file-from-a-form-with-axios var formData = ...

  5. python正则表达式函数match()和search()的区别详解

    match()和search()都是python中的正则匹配函数,那这两个函数有何区别呢? match()函数只检测RE是不是在string的开始位置匹配, search()会扫描整个string查找 ...

  6. TCP系列20—重传—10、早期重传(ER)

    一.介绍 在前面介绍thin stream时候我们介绍过有两种场景下可能不会产生足够的dup ACK来触发快速重传,一种是游戏类响应交互式tcp传输,另外一种是传输受到拥塞控制的限制,只能发送少量TC ...

  7. 3DMAX2016安装教程【图文】

    下载安装包之后,双击setup.exe. 下面是安装图片教程: 点击安装 点击下一步. 如图输入序列号和产品密钥. 填写安装路径,然后下一步. 开始安装,等待. 安装成功.

  8. 【Linux】- vi/vim

    所有的 Unix Like 系统都会内建 vi 文书编辑器,其他的文书编辑器则不一定会存在. 但是目前我们使用比较多的是 vim 编辑器. vim 具有程序编辑的能力,可以主动的以字体颜色辨别语法的正 ...

  9. java 文件操作知识点

    1.每个文件以一个文件路径和文件名称进行表示,在不同的操作系统环境下,文件路径的表示形式是不一样的,例如在Windows操作系统中一般的表示形式为C:\windows\system,而Unix上的表示 ...

  10. AndroidStudio3.0 注解报错Annotation processors must be explicitly declared now. The following dependencies on the compile classpath are found to contain annotation processor.

    把Androidstudio2.2的项目放到3.0里面去了,然后开始报错了. 体验最新版AndroidStudio3.0 Canary 8的时候,发现之前项目的butter knife报错,用到注解的 ...