使編譯器執(zhí)行預(yù)處理操作的代碼被稱為預(yù)處理指令,本文介紹常見的預(yù)處理指令的實際用法。
預(yù)處理符號是C語言內(nèi)置的符號,是可以直接使用的。
(資料圖片)
其中,若遵頊ANSI C,則__STDC__
為1,否則未定義。
#define可以用來定義標識符,其語法為:#define name stuff
,經(jīng)過預(yù)處理后,stuff
會被直接替換為·name
。
若stuff
的內(nèi)若過長,可在句末加上\
續(xù)行符號,像這樣:
#include#define Piccaso "Pablo,Diego,José\Francisco,de,Paula,Juan,Nepomuceno\,María,de,los,Remedios,Cipriano,de\,la,Santísima,Trinidad,Ruiz,y,Picasso"int main(){printf("%s", Piccaso);return 0;}
示例1:數(shù)值替換
int main(){int a = 100;return 0;}
示例2:循環(huán)替換
#includeint main(){while(1){printf("A");}return 0;}
運行代碼,將會在屏幕上死循環(huán)地打印A
。
示例3:分支替換
int main(){int input = 0;switch (input){case 1:break; case 2:break; case 3:}return 0;}
#define允許有參數(shù)的文本替換,這種操作通常稱為宏,其語法為:#define name(list) stuff
,其中,list
是由逗號隔開的符號表,符號有可能出現(xiàn)在stuff
中。
示例1:
int main(){printf("%d", 5+5);return 0;}
示例2:
int main(){printf("%d", 10*double(5+1));return 0;}
因為#define
的功能只是替換,若要利用宏定義實現(xiàn)快捷的函數(shù)操作,最好的方法是在宏定義時多加括號,以便于達到整體求值的效果,像這樣:#define double(x) (x)+(x)
。
注意:由于宏是直接替換,因此傳參時嚴禁使用自增,自減,傳參時使用,替換后依然會再次執(zhí)行,會導(dǎo)致不可預(yù)測的后果。
字符串有自動連接的特點,例如運行以下這段代碼:
#includeint main(){printf("123" "456");return 0;}
效果圖:
字符串轉(zhuǎn)換符#就是利用這個特性,它可以將宏定義中傳入的參數(shù),替換為字符串格式。
#include#define sum(x) printf("the val of "#x" is %d",x)int main(){int a = 10;sum(a);return 0;}
在上述代碼中,#號
將a
直接轉(zhuǎn)化為字符串,隨后三個字符串拼接在一起。
效果圖:
利用該方法可以只傳參一次實現(xiàn)值和名
同時打印。
在宏定義時,片段連接符##
可以實現(xiàn)將兩個符號連接在一起,使其成為一個符號,前提是這個合成的符號必須已經(jīng)被定義。
#include#define double(x) sum##x*=2int main(){int sum1 = 1;int sum2 = 1;int sum3 = 1;double(1);printf("%d %d %d", sum1, sum2, sum3);return 0;}
在上述代碼中,##
會把sum和參數(shù)x連接在一起,當我們傳入1經(jīng)過預(yù)處理后,等效于:sum1*=2
。
效果圖:
宏定義的優(yōu)勢:
宏定義的執(zhí)行速度遠遠超過函數(shù),當執(zhí)行簡單的計算時,更適合使用宏定義。宏定義傳參時沒有類型檢測,可以將任意的數(shù)據(jù)傳入。宏定義是直接替換,可以傳入各種各樣的符號,實現(xiàn)許許多多函數(shù)做不到的功能。(可以傳入類型、傳入函數(shù)、傳入語句等等)宏定義的劣勢:
宏定義不能調(diào)試、不能遞歸,因此宏定義只適合做簡單的計算。宏定義是直接替換,因此相鄰操作符的優(yōu)先級很有可能產(chǎn)生不期望的順序,因此要盡可能帶括號。宏定義傳參沒有類型檢測,因此不夠嚴謹。以下幾條公約,必須遵守
宏定義的名必須全部大寫。函數(shù)名不可以全部大寫。三、#undef#undef
宏定義刪除,可以在函數(shù)內(nèi)部使用!被刪除后的標識就不能再使用了。
指在VScode
或Linux
等用命令行執(zhí)行編譯的環(huán)境下,可以在編譯時對變量進行賦值。
在寫程序時,有些代碼是為了查看某個部分是否正確而寫的的調(diào)試代碼。
刪除很可惜,但又不想讓其編譯,此時就可以使用選擇性編譯
。
但實質(zhì)上使用if
語句或直接注釋
會更加方便,但在C語言內(nèi)置的頭文件中,為了節(jié)約時間經(jīng)常使用條件編譯。
#if 常量表達式//...#endif
常量表達式為真,則中間的語句編譯;
常量表達式為假,則中間的語句不編譯。
此外,也可以寫成多分支的表達式條件編譯。
int main(){#if 0printf("111");#elif 1printf("222");#else 0printf("333");#endifreturn 0;}
效果圖:
判斷某個符號是否被定義,只要被定義,就編譯中間的語句,無論其被定義為什么。
#include#define MAXint main(){#if defined(MAX)//或#ifdef MAXprintf("111");#endifreturn 0;}
或判斷某個符號是否沒定義,沒定義則編譯。
#include#define MAXint main(){#if !defined(MAX)//或#ifndef MAXprintf("111");#endifreturn 0;}
條件編譯是可以互相嵌套的。
#include#define DEBUG int main(){#ifdef DEBUG#if 1printf("111");#elif 0printf("222");#endif#endif return 0;}
如上述代碼是在是否定義判斷中嵌套常量表達式判斷。
效果圖:
對于#include
來說,后面的文件有兩種引用方法:
所有的頭文件在包含時都可以使用雙引號,但為了速度和區(qū)別位置,建議自己寫的頭文件用雙引號,標準庫中的用尖括號。
可以將許許多多的頭文件都包含在一個自己創(chuàng)建的頭文件中,最后只需要在其他的源文件中包含該自己創(chuàng)建的頭文件即可,像這樣:
在寫多人合作的大型項目時,每個程序員可能都要包含一次公用的頭文件,當他們寫的代碼匯總時,這個頭文件可能會被包含多次。
因此,我們使用條件編譯來解決這個問題。
#if !defined(TIME)#define TIME //...//... //在這里實現(xiàn)各種函數(shù)//...#endif
假設(shè)上述代碼為head.h
,當我們第一次包含head.h
時,由于TIME沒有被定義,因此會定義一個TIME,同時編譯里面的函數(shù)。
當我們第二次包含head.h
時,因為TIME被定義過了,即使head.h
里面的內(nèi)容被拷貝到源文件中,也不會進行編譯,從而加快了速度。
注意:在頭文件開頭加入#pragma once
即可一鍵實現(xiàn)上述效果,不必冗雜的代碼,但僅限于自己寫的頭文件,標準庫的頭文件已經(jīng)幫你加完了。
感謝您的閱讀與耐心~
Copyright © 2015-2022 太平洋產(chǎn)業(yè)網(wǎng)版權(quán)所有 備案號:豫ICP備2022016495號-17 聯(lián)系郵箱:93 96 74 66 9@qq.com