当前位置:编程学习 > C#/ASP.NET >>

关于宏#define使用陷阱总结

宏定义发生在预编译阶段,简单的说本质就是文本替换。使用时,有以下注意事项:

1,用宏定义表达式时,要使用完备的括号

如一下三个例子:

#define ADD(a,b) a+b

#define ADD(a, b) (a + b)

#define ADD(a, b)  (a) +(b)

这三种定义,全部都是不符合要求的。陷阱如下:

在计算ADD(a,b)*ADD(c,d)时,显然第一种出问题了。

#define MULTIPLE(a, b)  (a*b)  在计算(a+b)*c时,调用MULTIPLE(a+b,c)得到的结果错误。

因此一定要使用完备的括号,如下示例:

#define ADD(a,b) ((a) + (b))

2,使用宏定义,不允许参数发生变化,这也就是带参数的宏定义和函数的区别:

如下测试源码:

[cpp] 
<span style="font-size:18px;">#include <stdio.h> 
#define sqrt(a) ((a)*(a)) 
int fsqrt(int a) 

    return a*a; 

int main() 

     
    int a = 10, b = 10; 
    int r1, r2; 
     r1 = sqrt(a++); 
     r2 = fsqrt(b++); 
    printf("a = %d, b = %d, r1 = %d, r2 = %d\n", a, b, r1, r2); 
    return 0; 
}</span> 

最终的结果是a = 12; b = 11; r1 = 100; r2 = 100; 以上结果在vc6.0下获得。之所以a变成12,是因为在替换的时候,a++被执行了两次。要避免这种行为,就要使宏参数不发生变化。

如:a++; r1 = sqrt(a), 一切就ok了!

3,使用大括号将宏定义包含的多条表达式括起来。

如下示例:

[cpp] 
<span style="font-size:18px;">#include <stdio.h> 
#define INITIAL(a, b)\ 
    a = 0;\ 
    b = 0; 
int main() 

     
    int a[5], b[5] ; 
    int i; 
    for(i=0; i<5; i++) 
    INITIAL(a[i], b[i]); 
 
    printf("a = %d, b = %d\n", a[0], b[0]); 
    return 0; 
}</span> 

结果打印a是正常的,但打印b却是未初始化的结果。因为简单的文本替换,不能保证多条表达式都放到for循环体内。(这里,如果单独初始化一个变量,没有for循环时没有问题的。)上述的宏定义应改为:

[cpp] 
<span style="font-size:18px;">#define INITIAL(a, b)\ 
    {\ 
    a = 0;\ 
    b = 0;\ 
    }</span> 

注意这个“\”号哦,表示下面的一行和当前行在预编译时被认为在同一行。

最后,再简单对比下#define和 typedef的区别,#define发生在预编译阶段, typedef发生在编译阶段,可参考http://topic.csdn.net/t/20030810/14/2129718.html。更多细节上的区别,稍后再谈。

不对之处,大家多指正。

 

 

补充:软件开发 , C# ,
CopyRight © 2012 站长网 编程知识问答 www.zzzyk.com All Rights Reserved
部份技术文章来自网络,