大麥植髮婉之C語言巨集定義

2022-06-28 19:06:12 字數 2900 閱讀 5003

植髮多少?大麥植髮婉之c語言巨集定義

巨集(macro)是預處理命令的一種,它允許用乙個識別符號來表示乙個字串。先看乙個例子:

#include

#define n 100

int main()

執行結果:

120該示例中的語句int sum = 20 + n;n100代替了。

#define n 100就是巨集定義,n為巨集名,100是巨集的內容。在預處理階段,對程式中所有出現的「巨集名」,預處理器都會用巨集定義中的字串去代換,這稱為「巨集替換」或「巨集展開」。

巨集定義是由源程式中的巨集定義命令#define完成的,巨集替換是由預處理程式完成的。

巨集定義的一般形式為:

#define  巨集名  字串

#表示這是一條預處理命令,所有的預處理命令都以 # 開頭。define是預處理命令。巨集名是識別符號的一種,命名規則和識別符號相同。字串可以是數字、表示式、if 語句、函式等。

這裡所說的字串是一般意義上的字串行,不要和c語言中的字串等同,它不需要雙引號。
程式中反覆使用的表示式就可以使用巨集定義,例如:

#define m (n*n+3*n)
它的作用是指定識別符號m來表示(y*y+3*y)這個表示式。在編寫**時,所有出現 (y*y+3*y) 的地方都可以用 m 來表示,而對源程式編譯時,將先由預處理程式進行巨集代替,即用 (y*y+3*y) 去替換所有的巨集名 m,然後再進行編譯。

將上面的例子補充完整:

#include

#define m (n*n+3*n)

int main()

執行結果:

input a number: 10↙

sum=1560

程式的開頭首先定義了乙個巨集 m,它表示 (n*n+3*n) 這個表示式。在 9 行**中使用了巨集 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

這顯然是不正確的。所以進行巨集定義時要注意,應該保證在巨集替換之後不發生歧義。

對巨集定義的幾點說明

1) 巨集定義是用巨集名來表示乙個字串,在巨集展開時又以該字串取代巨集名,這只是一種簡單粗暴的替換。字串中可以含任何字元,它可以是常數、表示式、if 語句、函式等,預處理程式對它不作任何檢查,如有錯誤,只能在編譯已被巨集展開後的源程式時發現。

2) 巨集定義不是說明或語句,在行末不必加分號,如加上分號則連分號也一起替換。

3) 巨集定義必須寫在函式之外,其作用域為巨集定義命令起到源程式結束。如要終止其作用域可使用#undef命令。例如:

#define pi 3.14159

int main()

#undef pi

void func()

表示 pi 只在 main() 函式中有效,在 func() 中無效。

4) **中的巨集名如果被引號包圍,那麼預處理程式不對其作巨集代替,例如:

#include

#define ok 100

int main()

執行結果:

ok該例中定義巨集名 ok 表示 100,但在 printf 語句中 ok 被引號括起來,因此不作巨集替換,而作為字串處理。

5) 巨集定義允許巢狀,在巨集定義的字串中可以使用已經定義的巨集名,在巨集展開時由預處理程式層層代換。例如:

#define pi 3.1415926

#define s pi*y*y /* pi是已定義的巨集名*/

對語句:

printf("%f", s);
在巨集代換後變為:

printf("%f", 3.1415926*y*y);

6) 習慣上巨集名用大寫字母表示,以便於與變數區別。但也允許用小寫字母。

7) 可用巨集定義表示資料型別,使書寫方便。例如:

#define uint unsigned int
在程式中可用 uint 作變數說明:

uint a, b;
應注意用巨集定義表示資料型別和用 typedef 定義資料說明符的區別。巨集定義只是簡單的字串替換,由預處理器來處理;而 typedef 是在編譯階段由編譯器處理的,它並不是簡單的字串替換,而給原有的資料型別起乙個新的名字,將它作為一種新的資料型別。

請看下面的例子:

#define pin1 int *

typedef int *pin2; //也可以寫作typedef int (*pin2);

從形式上看這兩者相似, 但在實際使用中卻不相同。

下面用 pin1,pin2 說明變數時就可以看出它們的區別:

pin1 a, b;
在巨集代換後變成:

int * a, b;
表示 a 是指向整型的指標變數,而 b 是整型變數。然而:

pin2 a,b;
表示 a、b 都是指向整型的指標變數。因為 pin2 是乙個新的、完整的資料型別。由這個例子可見,巨集定義雖然也可表示資料型別, 但畢竟只是簡單的字串替換。在使用時要格外小心,以避出錯。

大麥植髮婉之C vector容器

植髮費用?大麥植髮婉之c vector容器 我們在宣告陣列的時候,採用的是datatype arrayname len 的形式,陣列在分配之後,不能調整大小,刪除和插入資料時操作十分的繁瑣,雖然可以採用鍊錶,但是鍊錶的操作更麻煩,我們希望有更簡單的方法。與string類一樣,向量vector 同屬於...

大麥植髮婉之C 類的繼承與派生

植髮多少?大麥植髮婉之c 類的繼承與派生 繼承是物件導向程式設計中最重要的乙個概念。繼承允許我們根據乙個類來定義另乙個類,達到了 功能重用效果。當建立乙個類時,如果待建立的類與另乙個類存在某些共同特徵,程式設計師不需要全部重新編寫成員變數和成員函式,只需指定繼承另乙個類即可,被繼承的類稱為基類或父類...

大麥植髮婉之C語言指標與二維陣列

植髮多少?大麥植髮婉之c語言指標與二維陣列 二維陣列在概念上是二維的,有行和列,但在記憶體中所有的陣列元素都是連續排列的,它們之間沒有 縫隙 以下面的二維陣列 a 為例 int a 3 4 從概念上理解,a 的分布像乙個矩陣 0 1 2 3 4 5 6 7 8 9 10 11 但在記憶體中,a 的分...