一、VA_LIST
1. VA_LIST 是在C语言中解决变参问题的一组宏 VA_LIST的用法:
(1)首先在函数里定义一具VA_LIST型的变量,这个变量是指向参数的指针
(2)然后用VA_START宏初始化变量刚定义的VA_LIST变量,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数。
(3)然后用VA_ARG返回可变的参数,VA_ARG的第二个参数是你要返回的参数的类型。
(4)最后用VA_END宏结束可变参数的获取。然后你就可以在函数里使用第二个参数了。如果函数有多个可变参数的,依次调用VA_ARG获取各个参数。
2. VA_LIST在编译器中的处理:
(1)在运行VA_START(ap,v)以后,ap指向第一个可变参数在堆栈的地址。
(2)VA_ARG()取得类型t的可变参数值,在这步操作中首先apt = sizeof(t类型),让ap指向下一个参数的地址。然后返回ap-sizeof(t类型)的t类型*指针,这正是第一个可变参数在堆栈里的地址。然后用*取得这个地址的内容。
(3)VA_END(),X86平台定义为ap = ((char*)0),使ap不再指向堆栈,而是跟NULL一样,有些直接定义为((void*)0),这样编译器不会为VA_END产生代码,例如gcc在Linux的X86平台就是这样定义的。
要注意的是:由于参数的地址用于VA_START宏,所以参数不能声明为寄存器变量,或作为函数或数组类型。
3.使用VA_LIST应该注意的问题:
(1)因为va_start, va_arg, va_end等定义成宏,所以它显得很愚蠢,可变参数的类型和个数完全在该函数中由程序代码控制,它并不能智能地识别不同参数的个数和类型. 也就是说,你想实现智能识别可变参数的话是要通过在自己的程序里作判断来实现的.
(2)另外有一个问题,因为编译器对可变参数的函数的原型检查不够严格,对编程查错不利.不利于我们写出高质量的代码。
小结:可变参数的函数原理其实很简单,而VA系列是以宏定义来定义的,实现跟堆栈相关。我们写一个可变参数的C函数时,有利也有弊,所 以在不必要的场合,我们无需用到可变参数,如果在C++里,我们应该利用C++多态性来实现可变参数的功能,尽量避免用C语言的方式来实现。 示例代码:
#include<stdio.h> #include<stdarg.h> void simple_va_fun(int start,...) { va_list arg_ptr; int value = start; int count = 0; va_start(arg_ptr,start); do { ++count; printf("the %d the arg:%d\n",count,value); value=va_arg(arg_ptr,int); }while(value != -1); return; } main() { simple_va_fun(100,-1); simple_va_fun(100,200,-1); }
测试结果:
the 1 the arg:100 the 1 the arg:100 the 2 the arg:200
出自:http://hi.baidu.com/kangliang/item/aa6ba5a94e82299f151073bd
二、syslog的配置信息
1. /var/log/messages包含所有的系统消息;/var/log/mail包含来自邮件系统的其它日志消息……
这些都可以在/etc/syslog.conf中进行配置
如果不小心把/var/log/messages删了,而重新touch messages后新的系统日志信息无法写入新创建的文件,可以通过执行logrotate -f /etc/logrotate.conf得以解决
2. 在使用过程中如果只答应出INFO信息可以答应而DEBUG信息打印不到messages中,则可以
1)修改/etc/syslog.conf添加一行:user.debug /var/log/debugs
注:格式为facility.level file
Facility Description
#define LOG_EMERG 0 /* system is unusable */
#define LOG_ALERT 1 /* action must be taken immediately */
#define LOG_CRIT 2 /* critical conditions */
#define LOG_ERR 3 /* error conditions */
#define LOG_WARNING 4 /* warning conditions */
#define LOG_NOTICE 5 /* normal but significant condition */
#define LOG_INFO 6 /* informational */
#define LOG_DEBUG 7 /* debug-level messages */
Priority Level Description
LOG_EMERG 紧急状况
LOG_ALERT 高优先级故障,如数据库损坏
LOG_CRIT 关键性(critical)错误,如硬件操作失败
LOG_ERR 一般错误
LOG_WARNING 一般警告
LOG_NOTICE 需要注意的特殊条件
LOG_INFO 通知性消息(Information messages)
LOG_DEBUG 调试消息
File:把对应的信息打印到哪个文件2)在程序中添加
static int log_upto = LOG_DEBUG; if (log_upto >= 0) setlogmask(LOG_UPTO(log_upto));
3)logrotate -f /etc/logrotate.conf
link: http://linux.die.net/man/3/syslog
http://www.groad.net/bbs/simple/?t612.html