题目:
代码如下 | 复制代码 |
#define N 2 #define M N+1 #define NUM 2*M+1 main( ) { int i; for(i=1;i<=NUM;i++) printf(“%dn”,i); } 该程序中的for循环执行的次数是( ) |
答案是6,跑过以后怎么都不理解,这是因为忽略了#define的重大不同。
#define进行的其实是简单的代入, 换句话说,NUM去找了M的表达式,把M的表达式代入,得出2*N+1+1,接着再去找N是多少,结果是2*2+1+1,因此等于6,而变量的话,毫无疑问则是进行了运算之后得出了的结果。
补充
1.简单的define定义
代码如下 | 复制代码 |
#define MAXTIME 1000 |
一个简单的MAXTIME就定义好了,它代表1000,如果在程序里面写
if(i<MAXTIME){.........}
编译器在处理这个代码之前会对MAXTIME进行处理替换为1000。
这样的定义看起来类似于普通的常量定义CONST,但也有着不同,因为define的定义更像是简单的文本替换,而不是作为一个量来使用,这个问题在下面反映的尤为突出。
2.define的“函数定义”
define可以像函数那样接受一些参数,如下
代码如下 | 复制代码 |
#define max(x,y) (x)>(y)?(x):(y); |
这个定义就将返回两个数中较大的那个,看到了吗?因为这个“函数”没有类型检查,就好像一个函数模板似的,当然,它绝对没有模板那么安全就是了。可以作为一个简单的模板来使用而已。
但是这样做的话存在隐患,例子如下:
代码如下 | 复制代码 |
#define Add(a,b) a+b; |
在一般使用的时候是没有问题的,但是如果遇到如:c * Add(a,b) * d的时候就会出现问题,代数式的本意是a+b然后去和c,d相乘,但是因为使用了define(它只是一个简单的替换),所以式子实际上变成了
c*a + b*d
另外举一个例子:
代码如下 | 复制代码 |
#define pin (int*); pin a,b; |
本意是a和b都是int型指针,但是实际上变成int* a,b;
a是int型指针,而b是int型变量。
这是应该使用typedef来代替define,这样a和b就都是int型指针了。
所以我们在定义的时候,养成一个良好的习惯,建议所有的层次都要加括号。
3.宏的单行定义
代码如下 | 复制代码 |
#define A(x) T_##x #define B(x) #@x #define C(x) #x |
我们假设:x=1,则有:
A(1)------〉T_1
B(1)------〉'1'
C(1)------〉"1"
(这里参考了 hustli的文章)
3.define的多行定义
define可以替代多行的代码,例如MFC中的宏定义(非常的经典,虽然让人看了恶心)
代码如下 | 复制代码 |
#define MACRO(arg1, arg2) do { / /* declarations */ / stmt1; / stmt2; / /* ... */ / } while(0) /* (no trailing ; ) */ |
关键是要在每一个换行的时候加上一个"/"