C 的巨集和 C 的巨集

2021-08-03 04:57:23 字數 4186 閱讀 2316

總結一下c和c++的巨集。

大家經常使用的其實都是c中的巨集,但是c++都延續了下來,先回顧一下。

#define :定義巨集

#undef :取消巨集

#include :包含標頭檔案

#ifdef 如果巨集已經定義,則返回真

#ifndef 如果巨集沒有定義,則返回真

#if 如果條件為真,則執行下面的

#else 與#if互斥使用

#elif 相當於我們使用的else if

#endif 結束#if作用域

#error 遇到標準錯誤,則輸出錯誤

#program  使用標準方法,給編譯器傳送特殊的命令

常用的c巨集已經說完,其實也是一些廢話,太easy了。那我不管了,還是再囉嗦吧

ansi c還定義了許多巨集,我們可以在程式設計的時候使用。

__date__ 當前日期

__time__ 當前時間

__file__ 當前的檔名

__line__ 當前行號

__stdc__ 當編譯器以ansi標準編譯時則是1

格式還是自己操作出來看比較好。

而且c預處理還提供了各種運算子

1.巨集延續運算子 \

所以這裡建議如果換行,最好加上此符號。

2.標記貼上運算子 ##

這個還是例子比較好理解

#include

#define

tokenpaster(n

)printf

("token"

#n " = %d", token##n)

intmain

(void

)

這裡想必已經理解一些了。結果如何?

token34 = 40

這個例項演示了 token##n 會連線到 token34 中,在這裡,我們使用了

字串常量化運算子(#)

和標記貼上運算子(##)

define() 運算子

預處理器defined運算子是用在常量表示式中的,用來確定乙個識別符號是否已經使用 #define 定義過。如果指定的識別符號已定義,則值為真(非零)。如果指定的識別符號未定義,則值為假(零)。

而#define 的運算的用法就不列了,記住就是簡單的替換,而且注意括號使用就ok。

c的巨集說完了,那說說c++的吧。

c++的變參巨集這個看到網上有人說,還挺詳細,所以直接copy過來僅供參考:

在 gnu c 中,巨集可以接受可變數目的引數,就象函式一樣,例如:

#define pr_debug(fmt,arg...) \

printk(kern_debug fmt,##arg)

用可變引數巨集(variadic macros)傳遞可變參數列

你可能很熟悉在函式中使用可變參數列,如:

void printf(const char* format, …);

直到最近,可變參數列還是只能應用在真正的函式中,不能使用在巨集中。

c99編譯器標準終於改變了這種局面,它允許你可以定義可變引數巨集(variadic macros),這樣你就可以使用擁有可以變化的參數列的巨集。可變引數巨集就像下面這個樣子:

#define debug(…) printf(__va_args__)

預設號代表乙個可以變化的參數列。使用保留名 __va_args__ 把引數傳遞給巨集。當巨集的呼叫展開時,實際的引數就傳遞給 printf()了。例如:

debug(「y = %d\n」, y);

而處理器會把巨集的呼叫替換成:

printf(「y = %d\n」, y);

因為debug()是乙個可變引數巨集,你能在每一次呼叫中傳遞不同數目的引數:

debug(「test」); //乙個引數

可變引數巨集不被ansi/iso c++ 所正式支援。因此,你應當檢查你的編譯器,看它是否支援這項技術。

用gcc和c99的可變引數巨集, 更方便地列印除錯資訊

gcc的預處理提供的可變引數巨集定義真是好用: 

#ifdef debug

#define dbgprint(format,args...) \

fprintf(stderr, format, ##args)

#else

#define dbgprint(format,args...)

#endif

如此定義之後,**中就可以用dbgprint了,例如dbgprint("aaa %s", __file__);。感覺這個功能比較cool  :em11: 

下面是c99的方法: 

#define dgbmsg(fmt,...) \

printf(fmt,__va_args__)

新的c99規範支援了可變引數的巨集

具體使用如下:

以下內容為程式**:

#include #include

#define logstrings(fm, ...) printf(fm,__va_args__)

int main()  

但現在似乎只有gcc才支援。

可變引數的巨集裡的『##』操作說明

帶有可變引數的巨集(macros with a variable number of arguments)

在2023年版本的iso c 標準中,巨集可以象函式一樣,定義時可以帶有可變引數。巨集的語法和函式的語法類似。下面有個例子:

#define debug(format, ...) fprintf (stderr, format, __va_args__)

這裡,『…』指可變引數。這類巨集在被呼叫時,它(這裡指『…』)被表示成零個或多個符號,包括裡面的逗號,一直到到右括弧結束為止。當被呼叫時,在巨集體(macro body)中,那些符號串行集合將代替裡面的__va_args__識別符號。更多的資訊可以參考cpp手冊。

gcc始終支援複雜的巨集,它使用一種不同的語法從而可以使你可以給可變引數乙個名字,如同其它引數一樣。例如下面的例子:

#define debug(format, args...) fprintf (stderr, format, args)

這和上面舉的那個iso c定義的巨集例子是完全一樣的,但是這麼寫可讀性更強並且更容易進行描述。

gnu cpp還有兩種更複雜的巨集擴充套件,支援上面兩種格式的定義格式。

在標準c裡,你不能省略可變引數,但是你卻可以給它傳遞乙個空的引數。例如,下面的巨集呼叫在iso c裡是非法的,因為字串後面沒有逗號:

debug ("a message")

gnu cpp在這種情況下可以讓你完全的忽略可變引數。在上面的例子中,編譯器仍然會有問題(complain),因為巨集展開後,裡面的字串後面會有個多餘的逗號。

為了解決這個問題,cpp使用乙個特殊的『##』操作。書寫格式為:

#define debug(format, ...) fprintf (stderr, format, ## __va_args__)

這裡,如果可變引數被忽略或為空,『##』操作將使預處理器(preprocessor)去除掉它前面的那個逗號。如果你在巨集呼叫時,確實提供了一些可變引數,gnu cpp也會工作正常,它會把這些可變引數放到逗號的後面。象其它的pasted macro引數一樣,這些引數不是巨集的擴充套件。

怎樣寫引數個數可變的巨集

一種流行的技巧是用乙個單獨的用括弧括起來的的 ``引數" 定義和呼叫巨集, 引數在 巨集擴充套件的時候成為類似 printf() 那樣的函式的整個引數列表。

#define debug(args) (printf("debug: "), printf args)

if(n != 0) debug(("n is %d\n", n));

明顯的缺陷是呼叫者必須記住使用一對額外的括弧。

gcc 有乙個擴充套件可以讓函式式的巨集接受可變個數的引數。 但這不是標準。另一種 可能的解決方案是根據引數個數使用多個巨集 (debug1, debug2, 等等), 或者用 逗號玩個這樣的花招:

#define debug(args) (printf("debug: "), printf(args))

#define _ ,

debug("i = %d" _ i);

c99 引入了對引數個數可變的函式式巨集的正式支援。在巨集 ``原型" 的末尾加上符號 ... (就像在引數可變的函式定義中), 巨集定義中的偽巨集 __va_args__ 就會在呼叫是 替換成可變引數。

最後, 你總是可以使用真實的函式, 接受明確定義的可變引數

如果你需要替換巨集, 使用乙個 函式和乙個非函式式巨集, 如 #define printf myprintf。

C 巨集中的 和

rt,define func a a define funcs a,b a b 第乙個 a的作用是將巨集引數a變成字串,如下面例子中,將對應的版本號轉換為字串 define version major 1 define version minor 29 define version patch 2 ...

C 語言 巨集定義和巨集函式

在軟體開發過程中,經常有一些常用或者通用的功能或者 段,這些功能既可以寫成函式,也可以封裝成為巨集定義。那麼究竟是用函式好,還是巨集定義好?這就要求我們對二者進行合理的取捨。我們來看乙個例子,比較兩個數或者表示式大小,首先我們把它寫成巨集定義 define max a,b a b a b 其次,把它...

C巨集的用法

前幾天參加某公司的筆試,有一道題是用純c實現乙個泛型函式。鬱悶了好久用c 模板實現了。巨集有如下的特點 1.與const相比,巨集是在預編譯的時候完成的 2.define 只做簡單的替換,不做型別安全檢查 3.使用不當會引起很多問題 巨集的用法 1.簡單的巨集定義 define max 1024 2...