#include <stdio.h> #define N 100 int main(){ int sum = 20 + N; printf("%d\n", sum); return 0; }運行結果:
int sum = 20 + N;
,N
被100
代替了。#define N 100
就是宏定義,N
為宏名,100
是宏的內容。在編譯預處理時,對程序中所有出現的“宏名”,都用宏定義中的字符串去代換,這稱為“宏代換”或“宏展開”。#define
完成的,宏代換是由預處理程序完成的。#define 宏名 字符串
#
表示這是一條預處理命令,所有的預處理命令都以#開頭。define
是預處理命令。宏名
是標識符的一種,命名規則和標識符相同。字符串
可以是常數、表達式等。
這裡所說的字符串是一般意義上的字符序列,不要和C語言中的字符串等同,它不需要雙引號。程序中反復使用的表達式就可以使用宏定義,例如:
#define M (n*n+3*n)它的作用是指定標識符
M
來代替表達式(y*y+3*y)
。在編寫源程序時,所有的(y*y+3*y)都可由M代替,而對源程序編譯時,將先由預處理程序進行宏代換,即用(y*y+3*y)表達式去替換所有的宏名M,然後再進行編譯。#include <stdio.h> #define M (n*n+3*n) int main(){ int sum, n; printf("Input a number: "); scanf("%d", &n); sum = 3*M+4*M+5*M; printf("sum=%d\n", n); return 0; }運行結果:
sum=3*M+4*M+5*M
中作了宏調用。在預處理時經宏展開後該語句變為:
sum=3*(n*n+3*n)+4*(n*n+3*n)+5*(n*n+3*n);需要注意的是,在宏定義中表達式
(n*n+3*n)
兩邊的括號不能少,否則會發生錯誤。如當作以下定義後:
#difine M n*n+3*n在宏展開時將得到下述語句:
s=3*n*n+3*n+4*n*n+3*n+5*n*n+3*n;這相當於: 3n2+3n+4n2+3n+5n2+3n 這顯然是不正確的。所以進行宏定義時要注意,應該保證在宏代換之後不發生錯誤。
#undef
命令。例如:
#define PI 3.14159 int main(){ // Code return 0; } #undef PI void func(){ // Code }表示PI只在main函數中有效,在func中無效。
#include <stdio.h> #define OK 100 int main(){ printf("OK\n"); return 0; }運行結果:
#define PI 3.1415926 #define S PI*y*y /* PI是已定義的宏名*/對語句:
printf("%f", S);在宏代換後變為:
printf("%f", 3.1415926*y*y);
#define UINT unsigned int在程序中可用UINT作變量說明:
UINT a, b;應注意用宏定義表示數據類型和用typedef定義數據說明符的區別。宏定義只是簡單的字符串代換,是在預處理完成的,而typedef是在編譯時處理的,它不是作簡單的代換,而是對類型說明符重新命名。被命名的標識符具有類型定義說明的功能。
#define PIN1 int * typedef (int *) PIN2;從形式上看這兩者相似, 但在實際使用中卻不相同。
PIN1 a,b;在宏代換後變成:
int *a,b;表示a是指向整型的指針變量,而b是整型變量。然而:
PIN2 a,b;表示a、b都是指向整型的指針變量。因為PIN2是一個類型說明符。由這個例子可見,宏定義雖然也可表示數據類型, 但畢竟是作字符代換。在使用時要分外小心,以避出錯。