C语言在linux内核中do while妙用之法
Posted Engineer-Bruce_Yang
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C语言在linux内核中do while妙用之法相关的知识,希望对你有一定的参考价值。
为什么说do while(0) 妙?因为它的确就是妙,而且在linux内核中实现是相当的妙,我们来看看内核中的相关代码:
#define db_error(fmt, ...) \\
do { \\
fprintf(stderr, "(error): "); \\
fprintf(stderr, fmt, ##__VA_ARGS__); \\
} while (0)
这只是个普通的调试信息的输出,有人便会认为,你这不是多此一举吗?去掉do while(0)不一样也实现了吗?其实不然,我们看看例子就清楚了,尽管很简单:
int main(void)
{
while(0)
{
printf("hello world\\n");
}
do
{
printf("hello world1\\n");
}while(0);
return 0 ;
}
这是一段简单到不能再简单的代码了,但还是要提一下,请看运行结果:
谁都知道第一个while(0)肯定是不会运行的,因为while()括号中的数值等于0,逻辑判定为假,即代码块中的hello world不会运行,但是do while(0)就不一样了,do while(0)即使条件不成立,也会拼了老命的去执行一次!
也就是说,为什么内核代码要这样来做,这是因为内核代码采用do{}while(0);这种结构可以保证无论在什么地方都可以正确的执行一次 ,这就是它用得最妙的地方,否则有时候调试程序的时候,单单的调试语句写了没打印其实是很正常的事情,不知道大家写代码的时候有没有遇到过,反正我是遇到过了,后来就是用这样的一种方法定位到错误点,顺利改正。
代码虽简单,但是用好用精熟练使用不一定什么时候都能想得到,越简单的东西,有时候,适用价值还是很好的!
分享以下我实现的调试输出程序,以后可以拿来当模版开发了:
#include <stdio.h>
#include <stdarg.h>
//内核代码采用do{}while(0);这种结构可以保证无论在什么地方都可以正确的执行一次
#define db_error(fmt, ...) \\
do { \\
fprintf(stderr, "(error): "); \\
fprintf(stderr, fmt, ##__VA_ARGS__); \\
} while (0)
#define db_msg(fmt, ...) \\
do { \\
fprintf(stdout, "(msg): "); \\
fprintf(stdout, fmt, ##__VA_ARGS__); \\
} while (0)
#define db_warn(fmt, ...) \\
do { fprintf(stdout, "(warn): "); \\
fprintf(stdout, fmt, ##__VA_ARGS__); \\
} while (0)
#define db_debug(fmt, ...) \\
do { \\
fprintf(stdout, "(debug): "); \\
fprintf(stdout, fmt, ##__VA_ARGS__); \\
} while (0)
int main(void)
{
db_error("h\\n");
db_warn("e\\n");
db_debug("llo\\n");
return 0 ;
}
运行结果:
调试信息在前,很快就可以知道在什么地方打印的语句,方便DEBUG!迅速找到程序bug的定位!
以上是关于C语言在linux内核中do while妙用之法的主要内容,如果未能解决你的问题,请参考以下文章