【C++基础 03】do...while(0)妙用
我的主题是,有时候知道一些细节会让你写出更好的代码。
============================================
之前学coocs2d-x的时候,发现有非常多do...while(0)的写法,一開始不明确为什么如此写,(起不到循环的作用),然后找了一下资料,发现这东西用处还蛮多的。如今来总结一下:
do...while(0)的妙用。
1.避免goto的使用
比方说我们须要在函数中处理一些错误。遇到错误则退出函数。当然退出之前我们须要释放一下资源。比方以下这样子:
bool HelloWorld::init()
{
// 分配资源
int *p = new int;
bool bRet = true; // 运行并进行错误处理
bRet = func1();
if(!bRet)
{
delete p;
p = NULL;
return false;
} bRet = func2();
if(!bRet)
{
delete p;
p = NULL;
return false;
} bRet = func3();
if(!bRet)
{
delete p;
p = NULL;
return false;
} // .......... // 运行成功,释放资源并返回
delete p;
p = NULL;
return true;
}
这样写最大的问题就是代码冗余。每次添加一个操作,就要处理一次,不灵活。所以这时候哦我们想到了goto,然后上面的版本号变成例如以下:
bool HelloWorld::init()
{
// 分配资源
int *p = new int;
bool bRet = true; // 运行并进行错误处理
bRet = func1();
if(!bRet) goto error; bRet = func2();
if(!bRet) goto error; bRet = func3();
if(!bRet) goto error; // 运行成功。释放资源并返回
delete p;
p = NULL;
return true; error:
delete p;
p = NULL;
return false;
}
当然,goto这东西尽管灵活方便,可是非常危急(这就跟红颜祸水这是一个意思,咦。怎么扯上这个了?)所以大部分的书籍都会建议你尽量程序中不要使用这个东西。
所以我们能够使用do...while来消除它。
例如以下:
bool HelloWorld::init()
{
// 分配资源
int *p = new int;
bool bRet = true; do
{
// 运行并进行错误处理
bRet = func1();
if(!bRet) break; bRet = func2();
if(!bRet) break; bRet = func3();
if(!bRet) break; }while(0); // 运行成功,释放资源并返回
delete p;
p = NULL;
return bRet;
}
2.宏定义中的do...while(0)
常常我们能在宏定义中看到这货的影子,比方说cocos2d-x中的
#define CC_SAFE_DELETE(p) do { delete (p); (p) = nullptr; } while(0)
那么又没有循环,也仅仅运行了一次,这样子写有什么意义呢?
假如我们去掉do...while。例如以下:
#define CC_SAFE_DELETE(p) delete (p); (p) = nullptr; if (p != NULL)
CC_SAFE_DELETE(p); //上面会转换成这样子
if (p != NULL)
delete (p);
(p) = nullptr;
如上。第二句永远运行,明显不合意思。有人可能会说,假设再加个括号呢,比方:
#define CC_SAFE_DELETE(p) {delete (p); (p) = nullptr; }
//有可能碰到这样的情况
if (p != NULL)
CC_SAFE_DELETE(p);
else
//do else sth
//上面会转换成这样
if (p != NULL)
{
delete (p);
(p) = nullptr;
};
else
//do else sth
else直接就编译错误了。
或许你会说,我们代码的习惯是在每一个推断后面加上{}, 就不会有这样的问题了,也就不须要do...while了,如:
if(...)
{
}
else
{
}
诚然,这是一个好的,应该提倡的编程习惯。可是我们须要做的是让代码具有通用性,强壮性,因此我们更提倡do...while(0)的使用方法。
========================================
转载请注明出处:http://blog.csdn.net/shun_fzll/article/details/37776429
【C++基础 03】do...while(0)妙用的更多相关文章
- javaSE基础03
javaSE基础03 生活中常见的进制:十进制(0-9).星期(七进制(0-6)).时间(十二进制(0-11)).二十四进制(0-23) 进制之间的转换: 十进制转为二进制: 将十进制除以2,直到商为 ...
- 从零3D基础入门XNA 4.0(2)——模型和BasicEffect
[题外话] 上一篇文章介绍了3D开发基础与XNA开发程序的整体结构,以及使用Model类的Draw方法将模型绘制到屏幕上.本文接着上一篇文章继续,介绍XNA中模型的结构.BasicEffect的使用以 ...
- 03、NetCore2.0下Web应用之搭建最小框架
03.NetCore2.0下Web应用之搭建最小框架 这里我们不使用VS2017或者CLI命令的方式创建Asp.Net Core 2.0网页应用程序,而是完全手工的一点点搭建一个Web框架,以便更好的 ...
- day33-线程基础03
线程基础03 6.用户线程和守护线程 用户线程:也叫工作线程,当线程的任务执行完或者通知方法结束.平时用到的普通线程均是用户线程,当在Java程序中创建一个线程,它就被称为用户线程 守护线程(Daem ...
- 从零3D基础入门XNA 4.0(1)——3D开发基础
[题外话] 最近要做一个3D动画演示的程序,由于比较熟悉C#语言,再加上XNA对模型的支持比较好,故选择了XNA平台.不过从网上找到很多XNA的入门文章,发现大都需要一些3D基础,而我之前并没有接触过 ...
- javascript基础03
javascript基础03 1. 算术运算符 后增量/后减量运算符 ++ ,-- 比较运算符 ( >, <, >=, <=, ==, !=,===,!== ) 逻辑运算符( ...
- C语言在linux内核中do while(0)妙用之法
为什么说do while(0) 妙?因为它的确就是妙,而且在linux内核中实现是相当的妙,我们来看看内核中的相关代码: #define db_error(fmt, ...) \ do { \ fpr ...
- Thread.Sleep(0)妙用
Thread.Sleep(0)妙用 我们可能经常会用到 Thread.Sleep 函数来使线程挂起一段时间.那么你有没有正确的理解这个函数的用法呢?思考下面这两个问题: 假设现在是 2008-4-7 ...
- 零基础快速入门SpringBoot2.0 (一)
零基础快速入门SpringBoot2.0 (一) 一.SpringBoot2.x依赖环境和版本新特性说明 简介:讲解新版本依赖环境和springboot2新特性概述 1.依赖版本jdk8以上, Spr ...
- 零基础入门 实战mpvue2.0多端小程序框架
第1章 课程快速预览(必看!!!)在这一章节中,老师讲带领你快速预览课程整体.其中,涉及到为什么要做这么一门实战课程.制作一个小程序的完整流程是怎么样的,以及如何做项目的技术选型. 第2章 30 分钟 ...
随机推荐
- hdu 1348(凸包)
Wall Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submis ...
- 前端读者 | 别人写的css,你敢用吗?
本文来自@yeaseonzhang:链接:http://yeaseonzhang.github.io/2018/04/10/%E5%88%AB%E4%BA%BA%E5%86%99%E7%9A%84cs ...
- linux下redis的最佳实践(Master-Slave)
本文演示了redis在同一台linux上的安装及运行多个实例,并演示了主从复制,以及如何进行主从的切换. 1. 下载 $ wget http://download.redis.io/releases/ ...
- codeforces Round #440 A Search for Pretty Integers【hash/排序】
A. Search for Pretty Integers [题目链接]:http://codeforces.com/contest/872/problem/A time limit per test ...
- UVA12096 集合栈计算机(map和vector实现双射关系+集合的交并运算的STL)
题目大意: 对于一个以集合为元素的栈,初始时栈为空. 输入的命令有如下几种: PUSH:将空集{}压栈 DUP:将栈顶元素复制一份压入栈中 UNION:先进行两次弹栈,将获得的集合A和B取并集,将结果 ...
- Struts2遇到Caused by Action class not found这类问题怎么解决
刚才碰到个小坑,显示的错误是struts.xml找不到Action,明显不会有错的因为之前还是正常跑起来的,然后百度google搜了一通没解决,maven clean用了没反应还是照样报错. 最后在百 ...
- POJ 3246 Game(凸包)
[题目链接] http://poj.org/problem?id=3246 [题目大意] 给出一些点,请删去一个点,使得包围这些点用的线长最短 [题解] 去掉的点肯定是凸包上的点,所以枚举凸包上的点去 ...
- 【树上莫队】【带修莫队】【权值分块】bzoj1146 [CTSC2008]网络管理Network
#include<cstdio> #include<cstring> #include<algorithm> #include<cmath> using ...
- 【set】bzoj3715 [PA2014]Lustra
对每种属性开一个set,只要某个厂家符合该属性的最值,就加进set,最后判断是否有某个厂家在4个set里都存在即可. #include<cstdio> #include<set> ...
- Java高级架构师(一)第15节:带查询的分页、修改和删除页面
@RequestMapping(value="toList",method=RequestMethod.GET) public String toList(@ModelAttrib ...