C语言宏定义小结

转自http://blog.sina.com.cn/s/blog_628bebe401013tf5.html


一、简单宏定义
 

不带参数的宏定义:
    #define 宏名字符序列
    #defineFALSE 0
  带参数的宏定义:
    #define宏名(形参表) 字符序列
    #defineMAX(a,b) ( (a) > (b) ) ? (a) : (b)
  取消宏定义:
    #undef宏名
二、条件编译
  #if_#endif
  #if_#else_#endif
  #if_#elif_#endif
 
  #ifdef_#endif
  #ifdef_#else_#endif
  #ifndef_#endif
  #ifndef_#else_#endif
三、复杂宏定义及do{}while(0)的使用
  #define foo()do{}while(0)

  #define foo(x) /
  action1(); /
  action2();

  if(NULL == pPointer)
    foo();
 就会出现action1和action2不会同时被执行的情况,而这显然不是程序设计的目的。
以上的情况用单独的{}也可以实现,但是为什么一定要一个do{}while(0)呢,看以下代码:
#define switch(x,y) {int tmp; tmp="x";x=y;y=tmp;}

if(x>y)

  switch(x,y);

else       //error, parse error before else
  otheraction();
在把宏引入代码中,会多出一个分号,从而会报错。
//------------------------------------------------
使用do{….}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

{

}

现有一个例子:

#define PJ_LOG(level,arg) \

dosomething();\
if (level <= pj_log_get_level()) \
pj_log_wrapper_##level(arg);

现在假设有以下应用:

if(L==1)

PJ_LOG(L,"AAA");

宏转开为:

if(L==1)

dosomething();
if (1 <= pj_log_get_level())

pj_log_wrapper_1("AAA"); ;
显然if(L==1)只管到dosomething();而后面的

if (1 <= pj_log_get_level())

pj_log_wrapper_1("AAA"); ;

则成了独立的语句。

假如使用do{}while(0)语句块,进行宏定义:

#define PJ_LOG(level,arg)do{ \

dosomething();\
if (level <= pj_log_get_level()) \
pj_log_wrapper_##level(arg); \

}while(0)

上述应用转开后为:

if(L==1)

do{

dosomething();
if (1<= pj_log_get_level())

pj_log_wrapper_1("AAA");

}while(0);

这样避免了意外的麻烦。OK现在明白了很多C程序中奇怪的do{}while(0)宏定义了吧


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值