带参数的宏和函数很相似,但有本质上的区别:宏展开仅仅是字符串的替换,不会对表达式进行计算;宏在编译之前就被处理掉了,它没有机会参与编译,也不会占用内存。
而函数是一段可以重复使用的代码,会被编译,会给它分配内存,每次调用函数,就是执行这块内存中的代码。
例1. 用函数计算平方值
#include <stdio.h> int SQ(int y) { return ((y)*(y)); } int main() { int i=1; while(i<=5){ printf("%d^2 = %d\n", (i-1), SQ(i++)); } return 0; }
例2. 用函数计算平方值
#include <stdio.h> int SQ(int y) { return ((y)*(y)); } int main() { int i=1; while(i<=5){ printf ("i is equal to %d\n", i); printf("%d^2 = %d\n", (i-1), SQ(i++)); } return 0; }
运行结果:
1^2 = 1
2^2 = 4
3^2 = 9
4^2 = 16
5^2 = 25
例3. 用宏计算平方值
#include <stdio.h>
#define SQ(y) ((y)*(y))
int main()
{
int i=1;
while(i<=5){
printf("%d^2 = %d\n", i, SQ(i++));
}
return 0;
}
这样的结果是不对的,在不同的编译器上运行结果也不一样。在我们的CB编译器的结果是:
3^2 = 2
5^2 = 12
7^2 = 30
在示例1中,先把实参 i 传递给形参 y,然后再自增 1,这样每循环一次 i 的值增加 1,所以最终要循环 5 次。
在示例3中,宏调用只是简单的字符串替换,SQ(i++) 会被替换为 ((i++)*(i++)),这样每循环一次 i 的值增加 2,所以最终只循环 3 次。
在第一次循环时,由于 i 等于 1,其计算过程为: i 自增 两次变为 3,因此结果的第一个值就是3^2; 第一个Y是一,第二个Y是二,所以结果是2.
在第二次循环时,由于 i 等于 3,其计算过程为: i 自增 两次变为 5,因此结果的第一个值就是5^2; 第一个Y是3,第二个Y是4,所以结果是12.
在第三次循环时,由于 i 等于 5,其计算过程为: i 自增 两次变为 7,因此结果的第一个值就是7^2; 第一个Y是5,第二个Y是6,所以结果是30.
因为此时i已经等于7,不再满足循环条件,停止循环。
由此可见,宏和函数只是在形式上相似,本质上是完全不同的。
带参数的宏也可以用来定义多个语句,在宏调用时,把这些语句又替换到源程序中,请看下面的例子:
例4.带参数的宏也可以用来定义多个语句
#include <stdio.h>
#define SSSV(s1, s2, s3, v) s1 = length * width; s2 = length * height; s3 = width * height; v = width * length * height;
int main()
{
int length = 3, width = 4, height = 5, sa, sb, sc, vv;
SSSV(sa, sb, sc, vv);
printf("sa=%d, sb=%d, sc=%d, vv=%d\n", sa, sb, sc, vv);
return 0;
}
运行结果:
sa=12, sb=15, sc=20, vv=60