C/C++宏定义中 do{...}while(0)用法分析
扫描二维码
随时随地手机看文章
在许多C/C++宏定义中我们会看到宏代码包含在do while循环或者if else语句里,例如:
#define M(X) do { f(X); g(X); } while (0)
#define M(X) if (1) { f(X); g(X); } else
那么这种用法有什么好处呢?这种用法最大的好处是增强宏定义代码的鲁棒性,尽管使得程序稍稍变得更加复杂。为了加快项目开发进度,现在一个工程的开发越来越需要相互合作与沟通交流,不管是利用别人的代码库还是写一些头文件供别人调用,都需要程序能够按照编程者的意图运行。但是对于这里提到的宏定义来说,你并不知道别人会怎么使用你的宏,如果定义不合理就会出现一些问题,例如下面这个宏定义:
#define M(X) f(x); g(x)
如果你这样使用这个宏:
if (condition)
M(x);
else
F();
宏展开时会变成:
if (condition)
f(x); g(x);
else
F();
这样就会出现语法错误,因为else找不到对应的if语句。如果给宏添加一个大括号,则变成:
if (condition)
{f(x); g(x);};
else
F();
这样也会出现语法错误,因为大括号后多了分号。针对这个问题,可以利用do while(0)或者if else来解决,例如:
#define M(X) do { f(X); g(X); } while (0)
或者:
#define M(X) if (1) { f(X); g(X); } else
这样调用后宏展开成:
if (condition)
do { f(X); g(X); } while (0);
else
F();
或者:
if (condition)
if (1) { f(x); g(x); } else;
else
F();
这种方式利用了分号在代码中的作用,可以有效克服C预处理器的一些缺点。有些编程指南告诉你要减少使用预处理器,这里提到的一点就是他们担心的一个原因。当然对于这种情况,你也可以把M(x)单独定义为一个函数,而不是一个宏。最后需要注意的一点是宏定义要写在一行,跨行需要加反斜杠''。