码迷,mamicode.com
首页 > 编程语言 > 详细

关于C语言的一些trick

时间:2015-01-26 19:14:00      阅读:303      评论:0      收藏:0      [点我收藏+]

标签:

很多东西已经记不起来了,想到一点写一点,碰到一点写一点,慢慢累积。

关于#

#在宏定义中用于替换传入变量的字符,例如: #define whole_operation(n)   do { printf(#n "=%d\n", (n));} while(0);

调用whole_operation(5*6), 输出:5*6=30, 有助于增加输出的可读性。

关于##

##是c99中定义的用于粘连两个符号,标识符或参数。例如:#define name_index(index)  name_##index

调用name_index(1),则生成name_1变量, 所以很多时候##用于动态的调用标识符具有一定规律的函数,宏或者变量。

举个列子,如果现在有add_arg_1(), add_arg_2()两个函数,只有在运行时才知道调用哪个函数,那么可以使用如下代码:

#define call_add_arg(argc)    add_arg_##argc()

IPnet的log模块中,因为log级别不同而log级别的前几个标识符都是IPCOM_LOG_,所以采用IPCOM_LOG_##x的方式在运行是判断需要输出什么级别的log。

关于不定参数...

很久以前在防火墙上做log模块的时候,用到一个比较有意思的trick。因为log要接收不同模块的不同信息,但是每个模块都含有自己独特的信息,为了保证所有信息都能被log接受,当时用了不定参的函数作为log的接口log(msgid, ...),通过va_list ap; va_start(ap, firstarg); va_arg(ap, type);va_end(ap);的方式来接受传入的各个参数。事实上,大多数prinf也是通过这个方式实现。 但是这种方式的函数调用容易出问题, 函数不知道什么时候参数结束,有可能导致程序崩溃。

C99定义了__VA_ARGS__ 用于接受不定参数的宏:#define LOG(msgid, ...)  log(msgid, __VA_ARGS__, lastarg) 或者#define LOG(msgid, arg...)  log(msgid, arg, lastarg)  lastarg是预定义的用于标识结束的宏或变量, 若不用__VA_ARGS__,则需用arg...来替代。 这样,LOG函数就可以接受任意多1个以上参数而不需要关系什么时候结束。那么如果LOG调用时只有一个msgid参数,就会变成log(msgid,,lastarg),这时候就需要借助##的另一个作用,如果##,后面没有参数,那么逗号就会被省略,于是LOG函数的最终定义变成: #define LOG(msgid, ...)  log(msgid, ##__VA_ARGS__, lastarg) 

 

关于C语言的一些trick

标签:

原文地址:http://www.cnblogs.com/omiao/p/4250992.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!