do{}while(0)的使用
在C语言中使用宏时经常可以看到“#define X do{}while(0)”语句,那这样写的作用是什么呢?我从网上搜了搜,现在把答案整理如下:
1. 避免宏展开时的逻辑误会
如:#define wait_event(wq,condition)
do{ if(condition) break; __wait_event(wq,condition); }while(0)
这是一个奇怪的循环,它根本就只会运行一次,为什么不去掉外面的do{..}while结构呢?有人曾一度把它叫做“怪圈”。原来这也是非常巧妙的技巧。在工程中可能经常会引起麻烦,而上面的定义能够保证这些麻烦不会出现。下面是解释:
假设有这样一个宏定义
#define macro(condition) if(condition) dosomething();
现在在程序中这样使用这个宏:
if(temp)
macro(i);
else
doanotherthing();
一切看起来很正常,但是仔细想想。这个宏会展开成:
if(temp)
if(condition) dosomething();
else
doanotherthing();
这时的else不是与第一个if语句匹配,而是错误的与第二个if语句进行了匹配,编译通过了,但是运行的结果一定是错误的。
为了避免这个错误,使用do{….}while(0) 把它包裹起来,成为一个独立的语法单元,从而不会与上下文发生混淆。同时因为绝大多数的编译器都能够识别do{…}while(0)这种无用的循环并进行优化,所以使用这种方法也不会导致程序的性能降低。
2. 处理末尾分号问题
这样使用可以在使用宏时直接给后面加分号,这样可以使宏更像语句,使用起来更方便。
有人为了解决上面提到的逻辑误会而直接使用{}将内容括起来,这样固然避免了逻辑误会,但无法在使用时给后面加分号。
评论