do {…} while (0) 在宏定义中的作用

do {…} while (0) 在宏定义中的作用do while 0 在宏定义中的作用 if feral bar wolf baz wolf elsebin wolf

大家好,欢迎来到IT知识分享网。

#define __set_task_state(tsk, state_value) \ do { 
      (tsk)->state = (state_value); } while (0) 

do{…}while(0)在C中是唯一的构造程序,让你定义的宏总是以相同的方式工作,这样不管怎么使用宏(尤其在没有用大括号包围调用宏的语句),宏后面的分号也是相同的效果。
这句话听起来可能有些拗口,其实用一句话概括就是:
使用do{…}while(0)构造后的宏定义不会受到大括号、分号等的影响,总是会按你期望的方式调用运行。

例如:

#define foo(x) bar(x); baz(x) 

然后你可能这样调用:

foo(wolf); 

这将被宏扩展为:

bar(wolf); baz(wolf); 

这的确是我们期望的正确输出。
下面看看如果我们这样调用:

if (!feral) foo(wolf); 

那么扩展后可能就不是你所期望的结果。上面语句将扩展为:

if (!feral) bar(wolf); baz(wolf); 

显而易见,这是错误的,也是大家经常易犯的错误之一。
几乎在所有的情况下,期望多语句宏来达到正确的结果是不可能的。如果我们使用do{…}while(0)来重新定义宏,即:

#define foo(x) do { 
      bar(x); baz(x); } while (0) 

现在,该语句功能上等价于前者:
do能确保大括号里的逻辑能被执行,而while(0)能确保该逻辑只被执行一次,即:同没有循环时一样。
对于上面的if语句,将会被扩展为:

if (!feral) do { 
    bar(wolf); baz(wolf); } while (0); 

从语义上讲,它与下面的语句是等价的:

if (!feral) { 
    bar(wolf); baz(wolf); } 

看这里你可能感到迷惑不解了,为什么不用大括号直接把宏包围起来呢?为什么非得使用do/while(0)逻辑呢?

例如,我们用大括号来定义宏如下:

#define foo(x) { 
      bar(x); baz(x); } 

这对于上面举的if语句的确能被正确扩展,但是如果我们有下面的语句调用呢:

if (!feral) foo(wolf); else bin(wolf); 

宏扩展后将变成:

if (!feral) { 
    bar(wolf); baz(wolf); }; else bin(wolf); 

总结:Linux和其它代码库里的宏都用do/while(0)来包围执行逻辑,因为它能确保宏的行为总是相同的,而不管在调用代码中使用了多少分号和大括号。还能在编译时避免由空宏引起的警告

由于内核不同体系结构的限制,我们可能需要多次使用空宏。在编译的时候,这些空宏会产生警告,为了避免这种警告,我们可以使用do{…}while(0)来定义空宏。

#define EMPTYMICRO do{ 
     }while(0)  

这样在编译的时候就不会产生警告。

免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://haidsoft.com/146847.html

(0)
上一篇 2025-04-11 20:33
下一篇 2025-04-11 20:45

相关推荐

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注微信