[编程技术] do{...}while(0)的用法
作者:精品下载站 日期:2021-11-22 15:00:51 浏览:35 分类:编程开发
零.导引
第一次见到 do{...}while(0)
是在学习 libevent 的时候,看到里面有很多类似
1
2
3
4
5
6
7
#define TT_URI(want) do { \
char *ret = evhttp_uri_join(uri, url_tmp, sizeof(url_tmp)); \
tt_want(ret != NULL); \
tt_want(ret == url_tmp); \
if (strcmp(ret,want) != 0) \
TT_FAIL(("\"%s\" != \"%s\"",ret,want)); \
} while(0)
当时特别疑惑,do{...}while()
不是做循环的吗,类似 for,while 的语法,不过现实开发中,用 for 和 while 的比较多,do{...}while()
比较少了,算是比较不常用的语法。
但是在这里,这样的代码一看就不是一个循环,do{...}while()
表面上在这里一点意义都没有,那么为什么要这么用呢?特别疑惑的google之,恍然大悟,原来 do{...}while()
还有此等妙用,看来自己还差得远啊。
总体来说,do{...}while(0)
有两种用法。
一.定义宏,实现局部作用域
大家做c语言题目的时候,一道必考题就是 #define 的算术运算。
比如,我随手写一个最简单的 #define
1
2
3
#define FUNC(x) x*3+4
...
int result = 2 * FUNC(3);
result输出多少? 26? 错!
这是c语言新手一定会犯的错误,至少我上大学的时候第一次看到这,我就做错了。
要知道这道题答案是多少,首先就要知道 #define 的作用。
- #define M (a+b) 它的作用是指定标识符M来代替表达式(a+b)。在编写源程序时,所有的(a+b)都可由M代替,而对源程序作编译时,将先由预处理程序进行宏代换,即用(a+b)表达式去置换所有的宏名M,然后再进行编译。
- c语言允许宏带有参数。在宏定义中的参数称为形式参数,在宏调用中的参数称为实际参数。对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。
也就是#define是在预处理的时候进行直接替换 (这句话是这一节的重点) 例如之上的展开就是:
int result = 2 * x * 3 + 4
x用实参3代替就是:
int result = 2 * 3 * 3 + 4 = 22
而不是 26.
有些人可能说,这些我都知道,这跟 do{…}while(0) 有什么关系。
其实,我只是为了告诉你,#define使用的时候要特别小心,尤其是#define一个很复杂的逻辑的时候。
我们举个简单的#define的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
void print()
{
cout<<"print: "<<endl;
}
void send()
{
cout <<"send: "<<endl;
}
#define LOG print();send();
int main(){
if (false)
LOG
cout <<"hello world"<<endl;
system("pause");
return 0;
}
这个代码输出什么?
理论上,if(false)
里面的代码不会被执行,也就是LOG不会被执行,所以只应该打印出 hello world
.
但是事实上:
1
2
send:
hello world
纳闷?
注意我上面说的一句话:
也就是 #define 是在预处理的时候进行直接替换!(这句话是这一节的重点)
也就是说,上面的 if(false)… 在这里是:
1
2
3
4
5
if (false)
print();
send();
cout <<"hello world"<<endl;
懂了吧。
怎么解决了,有些人马上想到,用 {…} 把 #define 的值括住不就可以了。的确,在这里是可以的。
我们在写代码的时候都习惯在语句右面加上分号,如果在宏中使用{},我们通常会这么写:
1
#define LOG {print();send();};
当我们的if后面有一个else呢?
就变成了:
1
2
3
4
5
6
7
8
9
if (false)
{
print();
send();
};
else
{
cout <<"hello"<<endl;
}
这样就会因为if语句后面多加了个 ";" 而编译不通过。不要说你说,那我不加 ";" 那要是你开发一个大型项目的时候你自己也不知道你自己要不要加 **";"**了,你就会被自己给绕晕了,所以统一的规范很重要。
那么来我们的最终版本 do{...}while(0);
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#define LOG do{print();send();}while (0);
int main(){
if (false)
LOG
else
{
cout <<"hello"<<endl;
}
cout <<"hello world"<<endl;
system("pause");
return 0;
}
就相当于:
1
2
3
4
5
6
7
8
9
10
11
if (false)
do{
print();
send();
}while (0);
else
{
cout <<"hello"<<endl;
}
cout <<"hello world"<<endl;
用 do{...}while(0);
包裹住要操作的 #define,无论你外面怎么操作,都不会影响 #define 的操作。妙哉妙哉啊。
二.替代goto
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
int dosomething()
{
return 0;
}
int clear()
{
}
int foo()
{
int error = dosomething();
if(error = 1)
{
goto END;
}
if(error = 2)
{
goto END;
}
END:
clear();
return 0;
}
当然这只是一个简单的例子,有些人说,我可以不用 goto,在每一个 goto 调用的地方直接,那么加一个判断,你就要加一条 clear(),万一你漏了呢?而且正常情况下, foo 里面的 if 有很多个,你要写很多 goto, END 里面的逻辑也更复杂。这样就更要小心。
由于 goto 不符合软件工程的结构化,而且有可能使得代码难懂,所以很多人都不倡导使用,那这个时候就可以用do{}while(0)
来进行统一的管理:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
int foo()
{
do
{
int error = dosomething();
if(error = 1)
{
break;
}
if(error = 2)
{
break;
}
} while (0);
clear();
return 0;
}
是不是看起来好看多了,而且还避免了由于错误导致的严重bug(比如你在clear里面是清理内存的操作,你忘记了写goto,而走不到END里面)。
在do{...}while(0)
里面,在任何地方都可以break跳出,然后继续下面的执行逻辑。即使你不写break,也会在执行完一遍do之后,while(0)不满足,自己跳出去。
<全文完>
猜你还喜欢
- 03-29 [编程相关] Winform窗体圆角以及描边完美解决方案
- 03-29 [前端问题] has been blocked by CORS policy跨域问题解决
- 03-29 [编程相关] GitHub Actions 入门教程
- 03-29 [编程探讨] CSS Grid 网格布局教程
- 10-12 [编程相关] python实现文件夹所有文件编码从GBK转为UTF8
- 10-11 [编程算法] opencv之霍夫变换:圆
- 10-11 [编程算法] OpenCV Camshift算法+目标跟踪源码
- 10-11 [Python] python 创建 Telnet 客户端
- 10-11 [编程相关] Python 基于 Yolov8 + CPU 实现物体检测
- 03-15 [脚本工具] 使用go语言开发自动化脚本 - 一键定场、抢购、预约、捡漏
- 01-08 [编程技术] 秒杀面试官系列 - Redis zset底层是怎么实现的
- 01-05 [编程技术] 《Redis设计与实现》pdf
取消回复欢迎 你 发表评论:
- 精品推荐!
-
- 最新文章
- 热门文章
- 热评文章
[短剧] 2025年06月03日 精选+付费短剧推荐25部
[软件合集] 25年6月3日 精选软件44个
[短剧合集] 2025年06月2日 精选+付费短剧推荐39部
[软件合集] 25年6月2日 精选软件18个
[软件合集] 25年6月1日 精选软件15个
[短剧合集] 2025年06月1日 精选+付费短剧推荐59部
[短剧] 2025年05月31日 精选+付费短剧推荐58部
[软件合集] 25年5月31日 精选软件66个
[电影] 黄沙漫天(2025) 4K.EDRMAX.杜比全景声 / 4K杜比视界/杜比全景声
[风口福利] 短视频红利新风口!炬焰创作者平台重磅激励来袭
[剧集] [央视][笑傲江湖][2001][DVD-RMVB][高清][40集全]李亚鹏、许晴、苗乙乙
[电视剧] 欢乐颂.5部全 (2016-2024)
[电视剧] [突围] [45集全] [WEB-MP4/每集1.5GB] [国语/内嵌中文字幕] [4K-2160P] [无水印]
[影视] 【稀有资源】香港老片 艺坛照妖镜之96应召名册 (1996)
[剧集] 神经风云(2023)(完结).4K
[剧集] [BT] [TVB] [黑夜彩虹(2003)] [全21集] [粤语中字] [TV-RMVB]
[资源] B站充电视频合集,包含多位重量级up主,全是大佬真金白银买来的~【99GB】
[影视] 内地绝版高清录像带 [mpg]
[书籍] 古今奇书禁书三教九流资料大合集 猎奇必备珍藏资源PDF版 1.14G
[美图] 2W美女个美女小姐姐,饱眼福
[电视剧] [突围] [45集全] [WEB-MP4/每集1.5GB] [国语/内嵌中文字幕] [4K-2160P] [无水印]
[剧集] [央视][笑傲江湖][2001][DVD-RMVB][高清][40集全]李亚鹏、许晴、苗乙乙
[电影] 美国队长4 4K原盘REMUX 杜比视界 内封简繁英双语字幕 49G
[电影] 死神来了(1-6)大合集!
[软件合集] 25年05月13日 精选软件16个
[精品软件] 25年05月15日 精选软件18个
[绝版资源] 南与北 第1-2季 合集 North and South (1985) /美国/豆瓣: 8.8[1080P][中文字幕]
[软件] 25年05月14日 精选软件57个
[短剧] 2025年05月14日 精选+付费短剧推荐39部
[短剧] 2025年05月15日 精选+付费短剧推荐36部
- 最新评论
-
- 热门tag