溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C語言預(yù)編譯的方法

發(fā)布時間:2022-04-27 10:36:56 來源:億速云 閱讀:160 作者:iii 欄目:開發(fā)技術(shù)

本篇內(nèi)容介紹了“C語言預(yù)編譯的方法”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!

一、預(yù)定義符號

預(yù)定義符號是系統(tǒng)本身定義的:

  • FILE 進行編譯的源文件的位置

  • LINE 文件當(dāng)前的行號

  • DATE 文件被編譯的日期

  • TIME 文件被編譯的時間

  • STDC 如果編譯器遵循 ASNSI C,其值為1,否者未定義

C語言預(yù)編譯的方法

二、#define 定義標(biāo)識符

語法:#define name stuff (用stuff替換name)

#define MAX 100   
#define STR "hehe"
int main()
{
	int max = MAX;
	printf("%d\n", max);  //輸出100
	printf("%s\n",STR);   //輸出 hehe
	return 0;
}

三、#define 定義宏

  • #define 機制包括了一個機制,允許把參數(shù)替換到文本中,這種實現(xiàn)通常稱為或者宏定義

  • 宏的申明方式:#define name(parament-list) stuff 其中的parament-list是一個由逗號隔開的符號表,他們可能出現(xiàn)在stuff中。

  • 注意:參數(shù)列表的左括號必須與name緊鄰,如果兩者之間有任何空白存在,參數(shù)列表就會解釋為stuff 的一部分。

#define SQUARE(X) X*X
int main()
{
	int ret = SQUARE(5);
	printf("%d\n",ret); //輸出25
	return 0;
}

上面的宏定義代碼存在一定的問題: 如果我們換一個參數(shù)(將5換成5+1)輸出的不是36而是11為什呢?

#define SQUARE(X) X*X
int main()
{
	int ret = SQUARE(5+1);//替換之后就是(5+1*5+1 = 11)
	printf("%d\n",ret);//輸出11
	return 0;
}

沒加括號
因此,用于對數(shù)值表達式進行求值的宏定義都應(yīng)該用這種方式加上括號,避免在使用宏時由于參數(shù)中的操作符或臨近操作符之間不可預(yù)料的相互作用。

正確的代碼:

#define SQUARE(X) (X)*(X))
int main()
{
	int ret = SQUARE(5+1);
	printf("%d\n",ret);//輸出36
	return 0;
}

四、#define 替換規(guī)則:

  • 一, #define NAME “l(fā)isa”
    程序中有"NAME",但”“內(nèi)的東西不會被宏替換。

  • 二,宏定義前面的那個必須是合法的用戶標(biāo)識符

  • 三,宏定義也不是說后面東西隨便寫,不能把字符串的兩個”“拆開。

  • 四: #define NAME “l(fā)isa”
    程序中有上面的宏定義,并且,程序里有句:
    NAMELIST這樣,不會被替換成"lisa"LIST

  • 五,宏不能出現(xiàn)遞歸

五、#和## 兩個符號(少見)

#的作用:把參數(shù)插入到字符串中

如果我們想要實現(xiàn)一個代碼:把參數(shù)插入到字符串中 用到“#”

C語言預(yù)編譯的方法

這里參數(shù)a,b就插入到了字符串中了
 ##的作用:可以把位于它兩邊的符號合成一個符號,它允許宏定義沖從分離的文本片段創(chuàng)建標(biāo)識符。

C語言預(yù)編譯的方法

圖中的三句代碼是等價的:

printf(“%d\n”,AGE(lisa,24));
printf(“%d\n”,AGE(lisa##24));
printf(“%d\n”,AGE(lisa24));

六、宏和函數(shù)的對比

函數(shù)和宏都能實現(xiàn)求兩個數(shù)的最大值

//函數(shù)
int Max(int x, int y)
{
	return (x > y ? x : y);
}
//宏
#define MAX(X,Y) ((X)>(Y)?(X):(Y))
int main()
{
	int a = 10;
	int b = 20;
	int max = Max(a, b); //輸出20
	printf("%d\n",max);
	max = MAX(a, b);
	printf("%d\n", max); //輸出20
	return 0;
}

通過分析上面的代碼實現(xiàn)用宏比用函數(shù)會更好,有兩個原因:

  • 用于調(diào)用函數(shù)和從函數(shù)返回的代碼可能比實際執(zhí)行的這個小型計算工作所需要的時間更多,所以宏比函數(shù)在程序的規(guī)模和速度方面更勝一籌。

  • 函數(shù)的參數(shù)必須申明為特定的類型。所以函數(shù)只能在類型合適的表達時式上使用。反之宏是與類型無關(guān)的。

當(dāng)然宏相比函數(shù)也有劣勢的地方:

  • 每次使用宏的時候,一份宏定義的代碼將替換插入到程序中。除非宏比較短,否者可能大幅度增加程序的長度。

  • 宏沒法調(diào)試

  • 宏由于類型無關(guān),也就不夠嚴謹

  • 宏可能會帶來運算符優(yōu)先級的問題,導(dǎo)致程序容易出錯。

#define定義宏和函數(shù)的對比表格

屬性#define定義宏函數(shù)
代碼長度每次使用時宏代碼都會被插入到程序中除了非常小的宏之外,程序的長度會大幅度增長函數(shù)的代碼只出現(xiàn)在一個地方,每次使用這個函數(shù)時,都調(diào)用那個地方的同一份代碼
執(zhí)行速度更快存在函數(shù)的調(diào)用和函數(shù)的額外開銷,所以速度相對慢一些
操作符優(yōu)先級宏參數(shù)的求值是在所有周圍表達式的上下文環(huán)境里,除非加上括號,否者鄰近操作符的優(yōu)先級可能產(chǎn)生不可預(yù)料的結(jié)果,所以建議宏在書寫的時候多用括號函數(shù)參數(shù)只在函數(shù)調(diào)用的時候求值一次,它的結(jié)果值傳遞給函數(shù)。表達式的求值結(jié)果更容易預(yù)測。
帶有副作用的參數(shù)參數(shù)可能被替換帶宏中的多個位置,所以帶有副作用的參數(shù)求值可能會產(chǎn)生不可預(yù)料的結(jié)果函數(shù)參數(shù)只在傳參的時候求值一次,結(jié)果更容易控制
參數(shù)類型宏的參數(shù)與類型無關(guān),只要對參數(shù)的操作是合法的,它就可以適用于任何參數(shù)類型函數(shù)的參數(shù)與類型有關(guān),如果參數(shù)的類型不同,就需要不同的函數(shù),即使他們執(zhí)行任務(wù)是不同的
調(diào)試和遞歸宏不方便調(diào)試,不能遞歸函數(shù)可以逐語句調(diào)試,可以遞歸

命名約定:把宏名全部大寫,函數(shù)名不要全部大寫。

七、#undef

#undef 指令用于移除一個宏定義

當(dāng)#undef 移除宏定義,再次使用報錯。如圖 :

C語言預(yù)編譯的方法

“C語言預(yù)編譯的方法”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI