一文帶你搞懂C語言預(yù)處理宏定義
預(yù)定義符號
這些預(yù)定義符號都是語言內(nèi)置的
__FILE__ //進(jìn)行編譯的源文件 __LINE__ //文件當(dāng)前的行號 __DATE__ //文件被編譯的日期 __TIME__ //文件被編譯的時間 __STDC__ //如果編譯器遵循ANSI C,其值為1,否則未定義
VS環(huán)境下未定義__STDC__ ,說明Visual Studio并未完全遵循ANSI C。
#define
#define 定義標(biāo)識符
#define name stuff //名稱;內(nèi)容
#define MAX 1000 #define reg register //為 register這個關(guān)鍵字,創(chuàng)建一個簡短的名字 #define do_forever for(;;) //用更形象的符號來替換一種實現(xiàn) #define CASE break;case //在寫case語句的時候自動把 break寫上。 // 如果定義的 stuff過長,可以分成幾行寫,除了最后一行外,每行的后面都加一個反斜杠(續(xù)行符)。 #define DEBUG_PRINT printf("file:%s\tline:%d\t \ date:%s\ttime:%s\n" ,\ __FILE__,__LINE__ , \ __DATE__,__TIME__ ) int main() { printf("%d\n", MAX); printf("%s\n", STR); do_forever; //執(zhí)行到這里會死循環(huán) return 0; }
在define進(jìn)行定義時,最好不要在后面加上分號 ;
,替換時也會將分號替換,容易導(dǎo)致問題。
#define 1000; int max = 0; if (3 > 1) max = MAX; else max = 0; //報錯
#define 定義宏
#define 機制包括了一個規(guī)定,允許把參數(shù)替換到文本中,這種實現(xiàn)通常稱為宏(macro)或定義宏(define macro)
#define name( parament-list ) stuff
其中的 parament-list 是一個由逗號隔開的符號表,它們可能出現(xiàn)在stuff中。
//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 = 5; int max = Max(a, b); printf("%d\n", max); return 0; }
最終得到較大值10。找到了宏的標(biāo)識,直接進(jìn)行符號替換。
- 參數(shù)列表的左括號必須與name緊鄰。
- 如果兩者之間有任何空白存在,參數(shù)列表就會被解釋為stuff的一部分
例如:
#define SQUARE(x) x*x int main() { printf("%d\n", SQUARE(5)); printf("%d\n", SQUARE(5 + 1)); //宏的參數(shù)是完全替換的,相當(dāng)于 5 + 1 * 5 + 1 結(jié)果為11 return 0; }
做改進(jìn):
#define SQUARE(x) (x)*(x) //這樣就不容易出錯 //最好最外層也加上括號
考慮如下計算:
#define Double(x) (x)+(x) int main() { int a = 5; printf("%d\n", (10 * Double(a))); //這里相當(dāng)于 10 * (5) + (5) 結(jié)果為55 return 0; }
在宏替換內(nèi)容上加上括號以保證得到預(yù)期的結(jié)果。
#define Double(x) ((x)+(x))
所以用于對數(shù)值表達(dá)式進(jìn)行求值的宏定義都應(yīng)該用這種方式加上括號,避免在使用宏時由于參數(shù)中的操作符或鄰近操作符之間不可預(yù)料的相互作用。
替換規(guī)則
在程序中擴展#define定義符號和宏時,需要涉及幾個步驟。
- 在調(diào)用宏時,首先對參數(shù)進(jìn)行檢查,看看是否包含任何由#define定義的符號。如果是,它們首先被替換。
- 替換文本隨后插入到程序中原來文本的位置。對于宏,參數(shù)名被他們的值所替換。
- 最后,再次對結(jié)果文件進(jìn)行掃描,看看它是否包含任何由#define定義的符號。如果是,就重復(fù)上述處理過程。
注意:
宏參數(shù)和#define 定義中可以出現(xiàn)其他#define定義的符號。但是對于宏,不能出現(xiàn)遞歸。
當(dāng)預(yù)處理器搜索#define定義的符號的時候,字符串常量的內(nèi)容并不被搜索。
# 和##
如何把參數(shù)插入到字符串中?
#define PRINT(FORMAT, VALUE) printf("the value of " #VALUE " is "FORMAT "\n", VALUE) //這里#可以把其后面的參數(shù)替換為字符串嵌入 int main() { printf("hello world\n"); printf("hello ""world\n"); //天然合為一個字符串(發(fā)現(xiàn)字符串是有自動連接的特點的) int a = 100; printf("The value a is %d\n", a); int b = 20; printf("The value b is %d\n", b); //那么考慮能否將同一個功能的打印操作合并 PRINT("%d", a); //這里只有當(dāng)字符串作為宏參數(shù)的時候才可以把字符串放在字符串中 /*代碼中的 #VALUE 會預(yù)處理器處理為: "VALUE" .*/ return 0; }
把宏對應(yīng)的參數(shù)替換為字符串
##的作用
將分離的片段合成為一個符號
#define CAT(A,B) A##B //把A和B組合成一個符號 int main() { int Windows11 = 2022; printf("%d\n", CAT(Windows, 11)); }
帶副作用的宏參數(shù)
當(dāng)宏參數(shù)在宏的定義中出現(xiàn)超過一次的時候,如果參數(shù)帶有副作用,那么你在使用這個宏的時候就可能出現(xiàn)危險,導(dǎo)致不可預(yù)測的后果。副作用就是表達(dá)式求值的時候出現(xiàn)的永久性效果。
x+1;//不帶副作用 x++;//帶有副作用
考慮宏:
//考慮宏的副作用 #define MAX(a, b) ( (a) > (b) ? (a) : (b) ) int main() { int x = 5; int y = 8; int z = MAX(x++, y++); printf("x=%d y=%d z=%d\n", x, y, z); //z = ((x++) > (y++) ? (x++) : (y++)); x = 6 y = 10 z = 9 }
到此這篇關(guān)于一文帶你搞懂C語言預(yù)處理宏定義的文章就介紹到這了,更多相關(guān)C語言預(yù)處理宏定義內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
iostream與iostream.h的區(qū)別詳細(xì)解析
以下是對C++中iostream與iostream.h的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來參考下2013-09-09C++ 中的INT_MAX,INT_MIN數(shù)值大小操作
這篇文章主要介紹了C++ 中的INT_MAX,INT_MIN數(shù)值大小操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-03-03如何通過wrap malloc定位C/C++的內(nèi)存泄漏問題
用C/C++開發(fā)的程序執(zhí)行效率很高,但卻經(jīng)常受到內(nèi)存泄漏的困擾。本文提供一種通過wrap malloc查找memory leak的思路。2021-05-05