帶參數的宏和函數很相似,但有本質上的區別:宏展開僅僅是字符串的替換,不會對表達式進行計算;宏在編譯之前就被處理掉了,它沒有機會參與編譯,也不會占用內存。而函數是一段可以重復使用的代碼,會被編譯,會給它分配內存,每次調用函數,就是執行這塊內存中的代碼。
【示例①】用函數計算平方值。
#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;
}
運行結果:
1^2 = 1
2^2 = 4
3^2 = 9
4^2 = 16
5^2 = 25
【示例②】用宏計算平方值。
#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;
}
在 VS2010 和 C-Free 5.0 下的運行結果:
3^2 = 1
5^2 = 9
7^2 = 25
在示例①中,先把實參 i 傳遞給形參 y,然後再自增 1,這樣每循環一次 i 的值增加 1,所以最終要循環 5 次。
在示例②中,宏調用只是簡單的字符串替換,SQ(i++) 會被替換為 ((i++)*(i++)),這樣每循環一次 i 的值增加 2,所以最終只循環 3 次。
由此可見,宏和函數只是在形式上相似,本質上是完全不同的。
帶參數的宏也可以用來定義多個語句,在宏調用時,把這些語句又替換到源程序中,請看下面的例子:
#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