[编程技术] do{...}while(0)的用法
作者:CC下载站 日期:2021-11-22 15:00:51 浏览:10 分类:编程开发
零.导引
第一次见到 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
取消回复欢迎 你 发表评论:
- 精品推荐!
-
- 最新文章
- 热门文章
- 热评文章
[壁纸工具] 每日必应壁纸爬取
[文件转换] Unlock Music 音乐解锁---搬运工---使用过最好的音乐格式转换全能工具
[图床工具] 百家号变免费图床
[行业软件] 超市播音系统V9.9(思维构造)——定时播放功能免费
[数码资讯] 松下Lumix S9完整细节曝光
[电视剧] [庆余年 第二季] [更至9集] [WEB-MKV] [国语中字] [4K]
[游戏娱乐] 《漫野奇谭》v1.16.535中文版
[游戏娱乐] 《奥西里斯:新黎明》v1.5.67中文版
[游戏娱乐] 《倾覆之国:最后一战》v1.0.0中文版
[文件传输] 【Android/IOS/Win】互传 EasyShare 3.6.5 零流量、极速、多平台快捷传输工具
[资料] [大学期末救急课] 猴博士+高斯课堂+斐多课堂,全集视频合集
[云资源] 价值2万元的老男孩Python教程
[书库] 史上最全摄影书推荐(附700本PDF版打包下载)
[云资源] 花了一千多元买的私人健身教程
[下载工具] Internet Download Manager 6.42.7 (IDM)
[影视] 灌篮高手 WEB-DL版下载/Slam Dunk/スラムダンク/灌篮高手:THE FIRST/灌篮高手电影版 2022 The First Slam Dunk 61.35G
[安卓软件] 酷我音乐APP_v10.7.6.4 去广告破解豪华VIP版
[即时通讯] 微信PC版WeChat 3.9.9.43 多开防撤回绿色版
[安卓软件] Solid Explorer文件管理器APP 2.8.38 破解版
[浏览器] Google Chrome v123.0.6312.59便携增强版
[云资源] 价值2万元的老男孩Python教程
[影视] 灌篮高手 WEB-DL版下载/Slam Dunk/スラムダンク/灌篮高手:THE FIRST/灌篮高手电影版 2022 The First Slam Dunk 61.35G
[云资源] 花了一千多元买的私人健身教程
[书库] 史上最全摄影书推荐(附700本PDF版打包下载)
[动画] 北斗神拳(1984) [两季合集] [MKV]
[资料] 抗战阵亡将士资料+续编
[电视剧] 三体 (2024) 全8集 网飞版本 中文字幕 合集
[纪录片] 河西走廊【10集 国语 中文字幕 1080P 10.8G MP4】
[电影] 2024年喜剧片·热辣滚烫 [mp4]
[影视] 铁爪 WEB-DL版下载 2023 The Iron Claw 23.48G
- 最新评论
-
我想看看mw2ddyy 评论于:04-26 好东西阿zfy123123 评论于:04-18 谢谢楼主xiaoqi 评论于:04-12 勿在线解压,勿手机解压,请在电脑上用最新款压缩软件解压!推荐360压缩或者好压CC下载站 评论于:04-10 无法解压啊,客服能不能给个解压教程ravengrey 评论于:04-10 谢谢支持!!CC下载站 评论于:03-26 很棒的资源,感谢分享云体风身 评论于:03-26 感谢分享,好东西云体风身 评论于:03-26 谢谢支持!CC下载站 评论于:03-14 央视精品,感谢付出提供。qwer9009 评论于:03-14
- 热门tag