在一些Linux内核和其它的开源代码中,我们经常看到像下面这样的代码:
1
2
3
|
do {
...
} while (0)
|
该代码片段并非循环,这样想想似乎使用do…while没有任何意义,那么为什么还要使用它呢?
实际上,do{...}while(0)的用途并不仅仅是优化你的代码。经过一系列的调研和探索,我们总结出它的一些用途如下。
1. 帮助定义复杂的宏以避免错误
详细解释请看“do {...} while (0) 在宏定义中的作用 ”,讲述了如何利用do {...}
while (0) 解决常犯的宏定义错误,让大家不再惧怕宏。
2. 避免使用goto控制程序流
在一些函数中,我们在return语句之前可能需要做一些工作,比如释放在函数一开始由malloc函数申请的内存空间,使用goto总是一种简单的方法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
int
foo()
{ somestruct* ptr = malloc (...);
dosomething...;
if (error)
{
goto
END;
}
dosomething...;
if (error)
{
goto
END;
}
dosomething...;
END: free (ptr);
return
0;
} |
但由于goto关键字可能会使代码不易读,因此许多人都不推荐使用它,那么我们可以使用do{...}while(0)来解决这一问题:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
int
foo()
{ somestruct* ptr = malloc (...);
do {
dosomething...;
if (error)
{
break ;
}
dosomething...;
if (error)
{
break ;
}
dosomething...;
} while (0);
free (ptr);
return
0;
} |
这里,我们使用do{...}while(0)来包含函数的主要部分,同时使用break替换goto,代码的可读性增强了。
3. 避免由宏引起的警告
由于内核不同体系结构的限制,我们可能需要多次使用空宏。在编译的时候,这些空宏会产生警告,为了避免这种警告,我们可以使用do{...}while(0)来定义空宏:
1
|
#define EMPTYMICRO do{}while(0) |
这样在编译的时候就不会产生警告。
4. 定义单一的函数块来完成复杂的操作
如果你有一个复杂的函数且你不想要创建新的函数,那么使用do{...}while(0),你可以将一些代码放在这里面并定义一些变量,这样你就不必担心do{...}while(0)外面的变量名是否与do{...}while(0)里面的变量名相同造成重复了。
如果你发现do{...}while(0)其它的好用途,请与我们分享哦!
编译自:http://www.pixelstech.net/article/1350871981-Significance-and-use-of-do%7B-%7Dwhile%280%29-