define中 與 的神奇用法

2021-06-21 01:18:21 字數 2493 閱讀 8036

2009-01-06 10:07:05

|  分類:

linux學習|舉報

|字型大小訂閱

本文整理自csdn。

#define f(a,b) a##b 

#define d(a) #a 

#define s(a) d(a) 

void main( void ) 

輸出結果: 

f(a,b) 

ab分析:  ##把兩個符號連起來 

#a指把a當成符號,就是把#後面的看成字串

# 和 ## 操作符是和#define巨集使用的. 使用# 使在#後的首個引數返回為乙個帶引號的字串. 例如, 命令 

#define to_string( s ) # s 

將會使編譯器把以下命令 

cout < < to_string( hello world! ) < < endl; 

理解為 

cout < < "hello world!" < < endl; 

使用##鏈結##前後的內容. 例如, 命令 

#define concatenate( x, y ) x ## y 

... 

int xy = 10; 

... 

將會使編譯器把 

cout < < concatenate( x, y ) < < endl; 

解釋為 

cout < < xy < < endl; 

理所當然,將會在標準輸出處顯示'10'.

puts(d(f(a,b)));  ----> 因為d巨集中的引數是另外乙個巨集,且帶##,所以作為引數的巨集不展開,相當於 

puts(#f(a,b));----->puts("f(a,b)"); 

puts(s(f(a,b))); ----> 因為s巨集中的引數是另外乙個巨集,但不帶##,所以作為引數的巨集先展開,相當於 

puts(s(ab));----->puts(d(ab));---->puts(#ab);---->puts("ab");

#define f(a,b) a##b 

#define d(a) #a --》 以"#"開頭的,直接替換,不展開:immediately replaced by the unexpanded actual argument 

#define s(a) d(a) --》 非以"#"開頭的,先展開,再替換,也就是一般的情況 

所以就兩種情況: 

1,不以"#"開頭的,先展開引數a,然後是替換**:puts(s(f(a,b)));-->puts(s(ab))-->puts(d(ab))-->puts("ab") 

2,以"#"開頭的,直接替換,不展開:puts(d(f(a,b)))-->puts("f(a,b)")

#include <

stdio.h

>

#define

direct_literal(a)  #a

#define

indirect_literal(a) direct_literal(a)

intmain(

void) 

這其實從編譯角度的展開歸約也可以理解啊。 

以上**第一種情況,當預編譯器看到direct_literal後查到它是巨集定義,定義為#a,此時後面的引數部分就會以#a的形式生成到原始檔中。也就是說,預編譯後的原始檔中,替代第一條語句的就是: 

puts("indirect_literal(a + b)");輸出則是indirect_literal(a + b) 

而對於第二條語句,當預編譯器看到indirect_literal後查到它是巨集定義,定義為direct_literal(a),這時先把它作為direct_literal(direct_literal(a + b))的形式暫存起來,你也可以理解為這個狀態是語法樹的當中乙個葉結點。然後再分析後面的direct_literal後面的引數部分,即:direct_literal(a + b),同樣,預編譯器會將它歸約為"a + b"的形式。這樣對於裡面的direct_literal(a + b)的形式就完全確定下來了,那麼這個值就可以充當葉子結點,即它底下不會再有結點。然後再回到剛才那個狀態,direct_literal("a + b")最後就是"\"a + b\""。所以這裡輸出是"a + b"。 

值得注意的是#a是將引數a轉為字串形式。所以像direct_literal(a)的展開形式是字串常量"a" 

那麼direct_literal("a")展開就是"\"a\""。 

#include 

<

stdio.h

>

#define

direct_literal(a)  #a

#define

indirect_literal(a) direct_literal(a)

#define

direct_cat(a, b)   a##b

#define

indirect_cat(a, b)  direct_cat(a, b)

intmain(

void) 

define中 與 的神奇用法

define f a,b a b define d a a define s a d a void main void 輸出結果 f a,b ab分析 把兩個符號連起來 a指把a當成符號,就是把 後面的看成字串 和 操作符是和 define巨集使用的.使用 使在 後的首個引數返回為乙個帶引號的字串....

define中 與 的神奇用法

define f a,b a b define d a a define s a d a void main void 輸出結果 f a,b ab分析 把兩個符號連起來 a指把a當成符號,就是把 後面的看成字串 和 操作符是和 define巨集使用的.使用 使在 後的首個引數返回為乙個帶引號的字串....

define中 與 的神奇用法

本文整理自csdn。define f a,b a b define d a a define s a d a void main void 輸出結果 f a,b ab分析 把兩個符號連起來 a指把a當成符號,就是把 後面的看成字串 和 操作符是和 define巨集使用的.使用 使在 後的首個引數返回...