**1 、为了宏展开的时候不会出错。如果直接放在花括号里会出错的
**
举例来说,假设你需要定义这样一个宏:
#define DOSOMETHING() action1(); action2();
这个宏的本意是,当执行DOSOMETHING()时,action1(),action2()都会被调用。如果有判断,再执行这个宏的话,如下:
if(NULL == pPointer)
DOSOMETHING();
else
...
//这样宏在预处理的时候会直接被展开,放在花括号里,那么实际上写的代码如下:if(NULL == pPointer)
action1();
action2();
else
…
这展开存在两个问题:因为if分支后面有两个语句,导致else分支没有对应的if,编译失败。假设没有else分支,则DOSOMETHING中的第二个语句无论if测试是否通过,都会执行。那么仅仅使用{}把action1()、action2()包起来行么?比如:
#define DOSOMETHING() { action1(); action2(); }
我们在写代码的时候都习惯在语句右面加上分号,如果在宏中使用{},代码编译展开后宏就相当于这样写了:{…};,展开后如下:
if(NULL == pPointer)
{
action1();
action2();
};
else
...
这段代码中大括号后多了一个分号,如果有else,那么else又没有对应的if了,编译出错。那么办法来了如果我们使用do{…}while(0)来定义宏,即:
`#define DOSOMETHING() \
do{ \
action1();\
action2();\
}while(0)\`
宏被展开后,上面的调用语句会保留初始的语义,同时绝大部分编译器都能够识别do{…}while(0)这种无用的循环并进行优化,不会导致性能优化的降低。
用一句话概括就是:“让你定义的宏总是以相同的方式工作,不管在调用代码中怎么使用分号和大括号,而该宏总能确保其行为是一致的。
2.避免使用GOTO,用break做跳出.
当你执行一段代码到一半,想跳过剩下的一半的时候,如果你正处于do while循环中,则能用break达到这个目的。