在C++中,有三种类型的循环语句:for, while, 和do...while, 但是在一般应用中作循环时, 我们可能用for和while要多一些,do...while相对不受重视。
    但是,最近在读我们项目的代码时,却发现了do...while的一些十分聪明的用法,不是用来做循环,而是用作其他来提高代码的健壮性。

1. do...while(0)消除goto语句。
通常,如果在一个函数中开始要分配一些资源,然后在中途执行过程中如果遇到错误则退出函数,当然,退出前先释放资源,我们的代码可能是这样:
version 1

bool Execute()
{
   // 分配资源
   int *p = new int;
   bool bOk(true);

// 执行并进行错误处理
   bOk = func1();
   if(!bOk) 
   {
      delete p;   
      p = NULL;
      return false;
   }

bOk = func2();
   if(!bOk) 
   {
      delete p;   
      p = NULL;
      return false;
   }

bOk = func3();
   if(!bOk) 
   {
      delete p;   
      p = NULL;
      return false;
   }

// ..........

// 执行成功,释放资源并返回
    delete p;   
    p = NULL;
    return true;
   
}

这里一个最大的问题就是代码的冗余,而且我每增加一个操作,就需要做相应的错误处理,非常不灵活。于是我们想到了goto:
version 2

bool Execute()
{
   // 分配资源
   int *p = new int;
   bool bOk(true);

// 执行并进行错误处理
   bOk = func1();
   if(!bOk) goto errorhandle;

bOk = func2();
   if(!bOk) goto errorhandle;

bOk = func3();
   if(!bOk) goto errorhandle;

// ..........

// 执行成功,释放资源并返回
    delete p;   
    p = NULL;
    return true;

errorhandle:
    delete p;   
    p = NULL;
    return false;
   
}

代码冗余是消除了,但是我们引入了C++中身份比较微妙的goto语句,虽然正确的使用goto可以大大提高程序的灵活性与简洁性,但太灵活的东西往往是很危险的,它会让我们的程序捉摸不定,那么怎么才能避免使用goto语句,又能消除代码冗余呢,请看do...while(0)循环:
version3

bool Execute()
{
   // 分配资源
   int *p = new int;

bool bOk(true);
   do
   {
      // 执行并进行错误处理
      bOk = func1();
      if(!bOk) break;

bOk = func2();
      if(!bOk) break;

bOk = func3();
      if(!bOk) break;

// ..........

}while(0);

// 释放资源
    delete p;   
    p = NULL;
    return bOk;
   
}

“漂亮!”, 看代码就行了,啥都不用说了...

2 宏定义中的do...while(0)
  如果你是C++程序员,我有理由相信你用过,或者接触过,至少听说过MFC, 在MFC的afx.h文件里面, 你会发现很多宏定义都是用了do...while(0)或do...while(false), 比如说:
#define AFXASSUME(cond)       do { bool __afx_condVal=!!(cond); ASSERT(__afx_condVal); __analysis_assume(__afx_condVal); } while(0)
粗看我们就会觉得很奇怪,既然循环里面只执行了一次,我要这个看似多余的do...while(0)有什么意义呢?
当然有!
为了看起来更清晰,这里用一个简单点的宏来演示:
#define SAFE_DELETE(p) do{ delete p; p = NULL} while(0)
假设这里去掉do...while(0),
#define SAFE_DELETE(p) delete p; p = NULL;
那么以下代码:
if(NULL != p) SAFE_DELETE(p)
else   ...do sth...
就有两个问题,
1) 因为if分支后有两个语句,else分支没有对应的if,编译失败
2) 假设没有else, SAFE_DELETE中的第二个语句无论if测试是否通过,会永远执行。
你可能发现,为了避免这两个问题,我不一定要用这个令人费解的do...while,  我直接用{}括起来就可以了
#define SAFE_DELETE(p) { delete p; p = NULL;}
的确,这样的话上面的问题是不存在了,但是我想对于C++程序员来讲,在每个语句后面加分号是一种约定俗成的习惯,这样的话,以下代码:
if(NULL != p) SAFE_DELETE(p);
else   ...do sth...
其else分支就无法通过编译了(原因同上),所以采用do...while(0)是做好的选择了。

也许你会说,我们代码的习惯是在每个判断后面加上{}, 就不会有这种问题了,也就不需要do...while了,如:
if(...)
{
}
else
{
}
诚然,这是一个好的,应该提倡的编程习惯,但一般这样的宏都是作为library的一部分出现的,而对于一个library的作者,他所要做的就是让其库具有通用性,强壮性,因此他不能有任何对库的使用者的假设,如其编码规范,技术水平等。

转载于:https://www.cnblogs.com/chengmin/archive/2010/01/22/1654151.html

do...while(0)的妙用相关推荐

  1. Sleep()简析 和Sleep(0)的妙用

    Sleep()简析 和Sleep(0)的妙用 原贴地址:残月:Thread.sleep(0)的意义 HawkJony:Sleep(0)的妙用 Thread.Sleep(0) 表示挂起0毫秒,你可能觉得 ...

  2. do while(0)的妙用

    do while(0)的妙用 do while(0);就如同一个花括号,具有独立的作用域,花括号所表示的符合语句是一个整体,do while(); 语句同样是一个整体,同样可以在if 等条件语句后直接 ...

  3. Sleep(0)的妙用

    Thread.Sleep(0) 表示挂起0毫秒,你可能觉得没作用,你要写Thread.Sleep(1000) 就有感觉了.似乎毫无意义. MSDN的说明:指定零 (0) 以指示应挂起此线程以使其他等待 ...

  4. Thread.Sleep(0)的妙用

    https://www.cnblogs.com/JianGuoWan/p/9139698.html Thread.Sleep(0) 表示挂起0毫秒,你可能觉得没作用,你要写Thread.Sleep(1 ...

  5. do...while(0)的妙用[转自:http://www.yuanma.org/data/2007/0209/article_2271.htm]

    1. do...while(0)消除goto语句. 通常,如果在一个函数中开始要分配一些资源,然后在中途执行过程中如果遇到错误则退出函数,当然,退出前先释放资源,我们的代码可能是这样: version ...

  6. 9亿做营销0.4亿做研发的妙可蓝多,靠什么卫冕“奶酪一哥“

    妙可蓝多最近刚交出一份亮眼财报. 3月24日,妙可蓝多发布的2021年财报显示,2021年妙可蓝多营收44.78亿元,同比增长57.31%:净利润1.54亿元,同比增长160.6%. 奇怪的是,财报发 ...

  7. 妙味课堂——JavaScript基础课程笔记

    集中时间把秒微课堂JS的基础课程看完,并且认真完成了课后练习.感觉在JS方面的技能算是入了个门了.课后练习的作业完成的代码我都汇总在了这里.至于视频课的学习笔记,则记录如下. 第01课JS入门基础_热 ...

  8. 估算带卷积核二分类0,3的网络的收敛时间和迭代次数

    制作一个网络分类minst的0和3求出这网络的迭代次数曲线表达式n(δ),和准确率表达式p-max(δ),用预期准确率去估算n,并推算需要的时间. 将minst的28*28的图片缩小到9*9,网络用一 ...

  9. 什么是 CNN?Facebook 员工写给小白的机器学习入门贴彻底火了!

    公众号关注 "视学算法" 设为 "星标",DLCV消息即可送达! 白交 发自 凹非寺  量子位 报道 | 公众号 QbitAI 看你是人还是物,是猫还是狗. 卷 ...

最新文章

  1. 链表一元多项式计算器的实现(Java语言描述)
  2. 单片机外文参考文献期刊_论文参考文献|如何在最短时间内加注参考文献??...
  3. 为什么物联网没有杀手级应用
  4. leetcode--200. 岛屿的个数
  5. [Buzz.Today]2013.03.14
  6. 联想gen系列服务器,Hpe Microserver Gen10 Plus开箱
  7. 如何在vmvare vsphere(ESXI)中移除磁盘中的文件
  8. dqn系列梳理_DQN是学习归纳出了策略,还是仅仅是memorize了大量的episodes?
  9. Vue学习笔记之09-v-model双向绑定
  10. 正则表达式批量重命名
  11. 电脑的眼泪--无限弹窗3
  12. ACM题库,分类整理
  13. EXIF 方向参数 Orientation
  14. SpringBoot使用Nacos作为配置中心服务和服务注册中心
  15. Paxos算法的一个简单小故事
  16. 中断调用与子程序调用
  17. PHP执行底层机制-zend详解
  18. iOS 判断是否为iPhoneX以上设备
  19. 沐风老师ATilesPro for 3dMax屋顶设计插件使用教程
  20. 模板的具体实现 ―― 模板的实例化(instantiation)或具体化(specialization)

热门文章

  1. 盘点物联网常用的八种通信协议
  2. 学会拐弯,才是人生大智慧(深度好文)
  3. 403 forbidden_[SpringSecurity] 自定义403页面
  4. 泰山服务器 oracle数据库,泰山之巅对话•Oracle数据库掌门人:领先对手10年?凭什么口气这么大?...
  5. vista下载_Vista和视图在游戏设计中的功能
  6. 隐马尔可夫模型(HMM)及Viterbi算法
  7. 台积电放大招:甩开英特尔 7nm和5nm芯片将诞生
  8. Thinkphp 验证码、文件上传
  9. unix环境高级编程基础知识之第二篇(3)
  10. [转]Android中pendingIntent的深入理解