C++/C宏定義中## 串連符與# 符的含義

來源:互聯網
上載者:User

http://blog.163.com/wshyao@126/blog/static/1070451420081018103237836/

## 串連符與# 符

  ## 串連符號由兩個井號組成,其功能是在帶參數的宏定義中將兩個子串(token)聯結起來,從而形成一個新的子串。但它不可以是第一個或者最後一個子串。所謂的子串(token)就是指編譯器能夠識別的最小文法單元。具體的定義在編譯原理裡有詳盡的解釋,但不知道也無所謂。同時值得注意的是#符是把傳遞過來的參數當成字串進行替代。下面來看看它們是怎樣工作的。這是MSDN上的一個例子。

  假設程式中已經定義了這樣一個帶參數的宏:

#define paster( n ) printf( "token" #n " = %d", token##n )

  同時又定義了一個整形變數:

int token9 = 9;

  現在在主程式中以下面的方式調用這個宏:

paster( 9 );

  那麼在編譯時間,上面的這句話被擴充為:

printf( "token" "9" " = %d", token9 );

  注意到在這個例子中,paster(9);中的這個”9”被原封不動的當成了一個字串,與”token”串連在了一起,從而成為了token9。而#n也被”9”所替代。

  可想而知,上面程式啟動並執行結果就是在螢幕上列印出token9=9 

 

http://kenshinf.blog.51cto.com/1088256/252541

關於記號粘貼操作符(token paste operator): ##1. 簡單的說,“##”是一種分隔串連方式,它的作用是先分隔,然後進行強制串連。   其中,分隔的作用類似於空格。我們知道在普通的宏定義中,前置處理器一般把空格
   解釋成分段標誌,對於每一段和前面比較,相同的就被替換。但是這樣做的結果是,
   被替換段之間存在一些空格。如果我們不希望出現這些空格,就可以通過添加一些
   ##來替代空格。   另外一些分隔標誌是,包括操作符,比如 +, -, *, /, [,], …,所以儘管下面的
   宏定義沒有空格,但是依然表達有意義的定義: define add(a, b)  a+b   而其強制串連的作用是,去掉和前面的字串之間的空格,而把兩者串連起來。2. 舉列 – 試比較下述幾個宏定義的區別   #define A1(name, type)  type name_##type##_type 或
   #define A2(name, type)  type name##_##type##_type   A1(a1, int);  /* 等價於: int name_int_type; */
   A2(a1, int);  /* 等價於: int a1_int_type;   */   解釋:
        1) 在第一個宏定義中,”name”和第一個”_”之間,以及第2個”_”和第二個
   ”type”之間沒有被分隔,所以前置處理器會把name_##type##_type解釋成3段:
   “name_”、“type”、以及“_type”,這中間只有“type”是在宏前面出現過
    的,所以它可以被宏替換。        2) 而在第二個宏定義中,“name”和第一個“_”之間也被分隔了,所以
   前置處理器會把name##_##type##_type解釋成4段:“name”、“_”、“type”
   以及“_type”,這其間,就有兩個可以被宏替換了。        3) A1和A2的定義也可以如下:
           #define A1(name, type)  type name_  ##type ##_type  
                                      <##前面隨意加上一些空格>
           #define A2(name, type)  type name ##_ ##type ##_type    結果是## 會把前面的空格去掉完成強串連,得到和上面結果相同的宏定義3. 其他相關 – 單獨的一個 #   至於單獨一個#,則表示 對這個變數替換後,再加雙引號引起來。比如      #define  __stringify_1(x)   #x
那麼
      __stringify_1(linux)   <==>  ”linux”所以,對於MODULE_DEVICE_TABLE     1) #define MODULE_DEVICE_TABLE(type,name)                        
             MODULE_GENERIC_TABLE(type##_device,name)
     2) #define MODULE_GENERIC_TABLE(gtype,name)                      
             extern const struct gtype##_id __mod_##gtype##_table     
             __attribute__ ((unused, alias(__stringify(name))))得到  
      MODULE_DEVICE_TABLE(usb, products)  
                             /*notes: struct usb_device_id products; */
 <==> MODULE_GENERIC_TABLE(usb_device,products)
 <==> extern const struct usb_device_id __mod_usb_device_table     
             __attribute__ ((unused, alias(”products”)))   注意到alias attribute需要一個雙引號,所以在這裡使用了__stringify(name)來
給name加上雙引號。另外,還注意到一個外部變數”__mod_usb_device_table”被alias
到了本驅動專用的由使用者自訂的變數products<usb_device_id類型>。這個外部變數
是如何使用的,更多的資訊請參看《probe()過程分析》。4. 分析方法和驗證方式 – 編寫一個簡單的C程式   用宏定義一個變數,同時用直接方式定義一個相同的變數,編譯報告重複定義;
   用宏定義一個變數,直接使用該宏定義的變數名稱,編譯通過且運行結果正確;
   使用printf列印字串資料。printf(”token macro is %s”, __stringify_1(a1));

 

 

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.