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

#define CC_BREAK_IF(cond)  if(cond) break
bool Excute()
{
bool bOk = false;
int* p = new int; do
{
CC_BREAK_IF(!isOk());
CC_BREAK_IF(!isOk1());
CC_BREAK_IF(!isOk2()); bOk = true;
} 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)

#define SAFE_DELETE_ARRAY(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

{

}

版权声明:本文博客原创文章。博客,未经同意,不得转载。

do...while(0)神奇的更多相关文章

  1. 诡异之--map clear 之后可能导致size != 0的操作

    map<char, int>mp; charMp[; charMp['b'] ++; cout<<charMp['a']<<endl; cout<<ch ...

  2. Javascript是单线程的深入分析

    本来想总结一下的,网上却发现有人已经解释的很清楚了,特转过来. 这也解释了为什么在用自动化测试工具来运行dumrendtree时设定的超时和测试case设定的超时的关联性. 面试的时候发现99%的童鞋 ...

  3. 【Tyvj 1060】【NOIP 2005】等价表达式

    设a为一个质数,模数为另一个质数,然后暴力算多项式的答案,如果答案相等就认为两个多项式相等. 这种hash有出错概率的题为什么还是要用hash呢?因为出错的概率实在太小了,a和模数的值取得好出题人根本 ...

  4. 【URAL 1486】Equal Squares

    题意:求给定字符矩阵中相同正方形矩阵的最大边长和这两个相同正方形的位置 第一次写字符串哈希,选两个不同的模数进行二维字符串哈希. 本来应该取模判断相等后再暴力扫矩阵来判断,但是我看到<Hash在 ...

  5. 单线程&浏览器多线程

    知乎答案:http://www.zhihu.com/question/31982417/answer/54136684 copy大牛的好文:from http://www.cnblogs.com/Ma ...

  6. 深入分析 Javascript 单线程

    面试的时候发现99%的童鞋不理解为什么JavaScript是单线程的却能让AJAX异步发送和回调请求,还有setTimeout也看起来像是多线程的?还有non-blocking IO, event l ...

  7. 单线程的JavaScript

    Javascript是单线程的 因为JS运行在浏览器中,是单线程的,每个window一个JS线程,既然是单线程的,在某个特定的时刻只有特定的代码能够被执行,并阻塞其它的代码.而浏览器是事件驱动的(Ev ...

  8. [USACO18DEC]Cowpatibility

    Description: Farmer John 的 \(N\) 头奶牛(\(2\le N\le 5\times 10^4\))各自列举了她们最喜欢的五种冰激凌口味的清单.为使这个清单更加精炼,每种可 ...

  9. JavaScript大杂烩17 - 性能优化

    在上一节推荐实践中其实很多方面是与效率有关的,但那些都是语言层次的优化,这一节偏重学习大的方面的优化,比如JavaScript脚本的组织,加载,压缩等等. 当然在此之前,分析一下浏览器的特征还是很有意 ...

随机推荐

  1. CMake入门(二)

    CMake入门(二) 最后更新日期:2014-04-25 by kagula 阅读前提:<CMake入门(一)>.Linux的基本操作 环境: Windows 8.1 64bit英文版.V ...

  2. 【Linux探索之旅】第二部分第一课:终端Terminal,好戏上场

    内容简介 1.第二部分第一课:终端Terminal,好戏上场 2.第二部分第二课预告:命令行,世界尽在掌握 终端Terminal,好戏上场 随着第一部分的结束,我们进入了第二部分(小编你这好像是废话. ...

  3. redis预切片技术,实现

    平时,我们将创建多个redis实例,以缓解单redis压力范例.但,作为高速缓存的数量增加.对redis对于扩展是一种非 - 不要无所谓.对redis有几种方法用于扩张可能.让我们添加每个redis最 ...

  4. Jquery简介选择的

    前言 Jquery一个js相框(程序代码相结合)这是一个程序开发过程中的半成品:分类似该框架EXTJS. 依赖库:jquery-XXX.js 语法:$() 正文 5择器 id选择器 $("# ...

  5. Singleton模式线程相关的(C\C++)

    这种需求的最新发展. 我需要一个静态类,无论地方,我可以在线程中调用它public功能对应的功能已经完成. 这个静态类会调用我初始化给它的一个指针,这个指针是与线程一一相应的: 准确来说这样的模式应该 ...

  6. crm创建基于fetch自己的自定义报告

    在解决方案资源管理器,右键点击"报表"目录.然后点击"增加了新的报告". 打开"报表向导". 在"欢迎来到报表向导"前, ...

  7. 小代码编写神器:LINQPad 使用入门

    原文:小代码编写神器:LINQPad 使用入门 一:概述 1:想查看程序运行结果,又不想启动 VS 怎么办? 2:想测试下自己的 C# 能力,不使用 VS 的智能感知,怎么办? 那么,我们有一个选择, ...

  8. .NET 4 并行(多核)编程系列之四 Task的休眠

    原文:.NET 4 并行(多核)编程系列之四 Task的休眠 .NET 4 并行(多核)编程系列之四 Task的休眠 前言:之前的几篇文章断断续续的介绍了Task的一些功能:创建,取消.本篇介绍Tas ...

  9. 最小二乘法拟合java实现源程序(转)

    因为我所在的项目要用到最小二乘法拟合,所有我抽时间将C++实现的程序改为JAVA实现,现在贴出来,供大家参考使用./** * <p>函数功能:最小二乘法曲线拟合</p> * @ ...

  10. [原创].NET 分布式架构开发实战之三 数据访问深入一点的思考

    原文:[原创].NET 分布式架构开发实战之三 数据访问深入一点的思考 .NET 分布式架构开发实战之三 数据访问深入一点的思考 前言:首先,感谢园子里的朋友对文章的支持,感谢大家,希望本系列的文章能 ...