浅谈do {...} while (0) 在宏定义中的作用

 更新时间:2020年4月25日 17:33  点击:1394

如果你是一名C程序员,你肯定很熟悉宏,它们非常强大,如果正确使用可以让你的工作事半功倍。然而,如果你在定义宏时很随意没有认真检查,那么它们可能使你发狂,浪费N多时间。在很多的C程序中,你可能会看到许多看起来不是那么直接的较特殊的宏定义。

下面就是一个例子:

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

在Linux内核和其它一些著名的C库中有许多使用do{...}while(0)的宏定义。这种宏的用途是什么?有什么好处?

Google的Robert Love(先前从事Linux内核开发)给我们解答如下:

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)的情况下。

如果我们使用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) 在宏定义中的作用就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持猪先飞。

[!--infotagslink--]

相关文章

  • C#中while循环语句用法实例详解

    这篇文章主要介绍了C#中while循环语句用法,以实例形式详细分析了while语句的用法,并对return,continue,break的区别做了进一步的分析,需要的朋友可以参考下...2020-06-25
  • shell脚本实战-while循环语句

    这篇文章主要介绍了shell脚本实战-while循环语句,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-12-09
  • Python While循环语句实例演示及原理解析

    这篇文章主要介绍了Python While循环语句实例演示及原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下...2020-04-30
  • 用while判断输入的数字是否回文数的简单实现

    这篇文章主要介绍了用while判断输入的数字是否回文数的简单实现,需要的朋友可以参考下...2020-04-25
  • 基于tensorflow for循环 while循环案例

    这篇文章主要介绍了基于tensorflow for循环 while循环案例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-06-30
  • PHP使用PDO操作数据库的乱码问题解决方法

    这篇文章主要介绍了PHP使用PDO操作数据库的乱码问题解决方法,实例分析了编码设置及属性设置解决乱码问题的实现技巧,需要的朋友可以参考下...2016-04-15
  • 使用do...while的方法输入一个月中所有的周日(实例代码)

    下面小编就为大家带来一篇使用do...while的方法输入一个月中所有的周日(实例代码)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2016-07-29
  • perl文件包含(do,require,use)指令介绍

    Perl中的文件包含,有三种方法:do, require, 以及use,这里简单的介绍下方便需要的朋友...2020-06-29
  • JavaScript中while循环的基础使用教程

    这篇文章主要给大家介绍了关于JavaScript中while循环的基础使用教程,文中通过示例代码介绍的非常详细,对大家学习或者使用JavaScript具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧...2020-08-11
  • Java while(scanner.hasNext())无法跳出的解决方案

    这篇文章主要介绍了Java while(scanner.hasNext())无法跳出的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...2021-10-30
  • 必升!Android O(安卓8.0)方便添加自定义铃声

    以往在 Android 系统里使用自定义铃声,要先将手机连接到电脑上,将歌曲和声音文件拖放到手机上,再将文件移动到 Ringtones 文件夹中才可以设置成功,处理起来相当麻烦,而且Android O(安卓8.0)方便添加自定义铃声,来看看吧...2017-07-06
  • php for while 与do while区别

    这是一个有味的问题哦,for ,while ,do while区别我想你认真仔细看了之后应该明白了它们三用法与区别了。 这是一个有味的问题哦,for ,while ,do while区别我想你认...2016-11-25
  • PHP - While Loop

    PHP - While Loop是php几种循环里面的一种,在很多数据我们循环读取数据时就会用到while loop语句哦,好了下面我就来看看吧. PHP - While Loop是php几种循环里面的...2016-11-25
  • 主流操作系统平台的宏定义

    这篇文章主要介绍了主流操作系统平台的宏定义,最近正在学习C++程序从Windows 平台向Unix 平台移植,参考了 qt 的宏定义文件,需要的朋友可以参考下...2020-04-25
  • C++中的while循环和for循环语句学习教程

    这篇文章主要介绍了C++中的while循环和for循环语句学习教程,是C++入门学习中的基础知识,需要的朋友可以参考下...2020-04-25
  • ADO.NET中的五个主要对象的详细介绍与应用

    ADO.NET中的五个主要对象:Connection、Command、DataAdapter DataSet、DataReader详细介绍与应用,感兴趣的朋友可以参考下...2021-09-22
  • C#中for循环、while循环循环执行的方法

    这篇文章主要介绍了C#中for循环、while循环循环执行的方法的相关资料,非常不错,具有参考借鉴价值,感兴趣的朋友一起学习吧...2020-06-25
  • python利用JMeter测试Tornado的多线程

    这篇文章主要介绍了python利用JMeter测试Tornado的多线程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-04-27
  • php实现基于PDO的预处理示例

    这篇文章主要介绍了php实现基于PDO的预处理,结合实例形式分析了php实现pdo预处理的相关操作技巧与注意事项,需要的朋友可以参考下...2017-04-03
  • 内联函数inline与宏定义深入解析

    类的内敛函数是一个真正的函数。使用内联函数inline可以完全取代表达式形式的宏定义...2020-04-25