c 語言中的巨集定義

2021-07-31 20:39:19 字數 3009 閱讀 5065

巨集定義

巨集定義是c提供的三種預處理功能的其中一種,這三種預處理包括:巨集定義、檔案包含、

條件編譯

1. 不帶引數的巨集定義:

巨集定義又稱為巨集代換、巨集替換,簡稱「巨集」。

格式: #define 識別符號 字串

其中的識別符號就是所謂的

符號常量,也稱為「巨集名」。

預處理(預編譯)工作也叫做巨集展開:將巨集名替換為字串。

掌握"巨集"概念的關鍵是「換」。一切以換為前提、做任何事情之前先要換,準確理解之前就要「換」。

例:   #define pi 3.1415926   把程式中出現的pi全部換成3.1415926

說明:(1)巨集名一般用大寫

(2)使用巨集可提高程式的通用性和易讀性,減少不一致性,減少輸入錯誤和便於修改。例如:陣列大小常用巨集定義

(3)預處理是在編譯之前的處理,而編譯工作的任務之一就是語法檢查,預處理不做語法檢查。

(4)巨集定義末尾不加分號;

(5)巨集定義寫在函式的花括號外邊,作用域為其後的程式,通常在檔案的最開頭。

(6)可以用#undef命令終止巨集定義的作用域

(7)巨集定義可以巢狀

(8)字串" "中永遠不包含巨集

(9)巨集定義不

分配記憶體,變數定義

分配記憶體。

2.  帶引數的巨集定義:

除了一般的字串替換,還要做引數代換

格式:   #define 巨集名(參數列) 字串

例如:#define s(a,b) a*b

area=s(3,2);第一步被換為area=a*b; ,第二步被換為area=3*2;

類似於函式呼叫,有乙個啞實結合的過程:

(1)實參如果是表示式容易出問題

#define s(r) r*r

area=s(a+b);第一步換為area=r*r;,第二步被換為area=a+b*a+b;

正確的巨集定義是#define s(r) ((r)*(r))

(2)巨集名和引數的括號間不能有空格

(3)巨集替換只作替換,不做計算,不做表示式求解

(4)函式呼叫在編譯後程式執行時進行,並且

分配記憶體。巨集替換在編譯前進行,不分配記憶體

(5)巨集的啞實結合不存在型別,也沒有型別轉換。

(6)函式只有乙個返回值,利用巨集則可以設法得到多個值

(7)巨集展開使源程式變長,函式呼叫不會

(8)巨集展開不佔執行時間,只佔編譯時間,函式呼叫佔執行時間(分配記憶體、保留現場、值傳遞、返回值)

3.  巨集定義其他冷門、重點知識

#define用法

1、 用無參巨集定義乙個簡單的常量

#define len 12

這個是最常見的用法,但也會出錯。

比如下面幾個知識點你會嗎?可以看下:

(1) #define name "zhangyuncong"

程式中有"name"則,它會不會被替換呢?

(2) #define 0x abcd

可以嗎?也就是說,可不可以用把識別符號的字母替換成別的東西?

(3) #define name "zhang

這個可以嗎?

(4) #define name "zhangyuncong"

程式中有上面的巨集定義,並且,程式裡有句:

namelist這樣,會不會被替換成"zhangyuncong"list

四個題答案都是否定的。

第乙個,""內的東西不會被巨集替換。這一點應該大都知道。

第二個,巨集定義前面的那個必須是合法的

使用者識別符號

第三個,巨集定義也不是說後面東西隨便寫,不能把字串的兩個""拆開。

第四個:只替換識別符號,不替換別的東西。namelist整體是個識別符號,而沒有name識別符號,所以不替換。

也就是說,這種情況下記住:#define 第一位置第二位置

(1) 不替換程式中字串裡的東西。

(2) 第一位置只能是合法的識別符號(可以是關鍵字)

(3) 第二位置如果有字串,必須把""配對。

(4) 只替換與第一位置完全相同的識別符號

還有就是老生常談的話:記住這是簡單的替換而已,不要在中間計算結果,一定要替換出表示式之後再算。

2、 帶參巨集一般用法

比如#define max(a,b) ((a)>(b)?(a):(b))

則遇到max(1+2,value)則會把它替換成:

((1+2)>(value)?(1+2):(value))

注意事項和無參巨集差不多。

但還是應注意

#define fun(a) "a"

則,輸入fun(345)會被替換成什麼?

其實,如果這麼寫,無論巨集的實參是什麼,都不會影響其被替換成"a"的命運。

也就是說,""內的字元不被當成形參,即使它和一模一樣。

那麼,你會問了,我要是想讓這裡輸入fun(345)它就替換成"345"該怎麼實現呢?

請看下面關於#的用法

3、 有參巨集定義中#的用法

#define str(str) #str

#用於把巨集定義中的引數兩端加上字串的""

比如,這裡str(my#name)會被替換成"my#name"

一般由任意字元都可以做形參,但以下情況會出錯:

str())這樣,編譯器不會把「)」當成str()的引數。

str(,)同上,編譯器不會把「,」當成str的引數。

str(a,b)如果實參過多,則編譯器會把多餘的引數捨去。(vc++2008為例)

str((a,b))會被解讀為實參為:(a,b),而不是被解讀為兩個實參,第乙個是(a第二個是b)。       4、 有參巨集定義中##的用法

#define wide(str) l##str

則會將形參str的前面加上l

比如:wide("abc")就會被替換成l"abc"

如果有#define fun(a,b) vo##a##b()

那麼fun(id ma,in)會被替換成void main()

5、 多行巨集定義:

#define doit(m,n) for(int i=0;i<(n);++i)\

C語言中的巨集定義

下列c 中包含兩種巨集定義,例如 include define max connection 1000 define mng port 5000 define min a,b a b a b define max a,b a b a b int main 方法1 gcc e 引數預編譯 gcc e ...

C語言中的巨集定義

c語言中的巨集定義 c語言中的巨集定義也叫做預處理命令,預處理命令是c語言本身的組成部分,不能直接對它們進行編譯。1.基本概念 1 預處理不是c語句,是在編譯前進行的 2 預處理功能主要用於下3種情況 巨集定義 檔案包含 條件編譯 3 巨集是在編譯前進行的,程式不能直接對它們進行編譯,由預處理程式先...

c語言解除巨集定義 C語言中巨集定義的用法

說到巨集定義,我們應該先了解什麼是預處理指令,相信大家並不會陌生,之前我們程式設計時,程式的開頭 include指令,define指令都是預處理指令,它能使我們的編譯更加的高效,便捷,因此c語言中是允許使用者自己加入一些特定的預處理指令的。一,不帶引數的巨集定義 不帶引數的巨集定義是比較簡單的,就是...