• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Error

            C++博客 首頁 新隨筆 聯系 聚合 管理
              217 Posts :: 61 Stories :: 32 Comments :: 0 Trackbacks

            C 宏

            1,防止一個頭文件被重復包含
            #ifndef BODYDEF_H
            #define BODYDEF_H
            //頭文件內容
            #endif
            2,得到指定地址上的一個字節或字
            #define MEM_B( x ) ( *( (byte *) (x) ) )
            #define MEM_W( x ) ( *( (word *) (x) ) )
            3,得到一個field在結構體(struct)中的偏移量
            #define FPOS( type, field ) ( (dword) &(( type *) 0)-> field )
            4,得到一個結構體中field所占用的字節數
            #define FSIZ( type, field ) sizeof( ((type *) 0)->field )
            5,得到一個變量的地址(word寬度)
            #define B_PTR( var ) ( (byte *) (void *) &(var) )
            #define W_PTR( var ) ( (word *) (void *) &(var) )
            6,將一個字母轉換為大寫
            #define UPCASE( c ) ( ((c) >= ''a'' && (c) <= ''z'') ? ((c) - 0x20) : (c) )
            7,判斷字符是不是10進值的數字
            #define DECCHK( c ) ((c) >= ''0'' && (c) <= ''9'')
            8,判斷字符是不是16進值的數字
            #define HEXCHK( c ) ( ((c) >= ''0'' && (c) <= ''9'') ||((c) >= ''A'' && (c) <= ''F'') ||((c) >= ''a'' && (c) <= ''f'') )
            9,防止溢出的一個方法
            #define INC_SAT( val ) (val = ((val)+1 > (val)) ? (val)+1 : (val))
            10,返回數組元素的個數
            #define ARR_SIZE( a ) ( sizeof( (a) ) / sizeof( (a[0]) ) )
            11,使用一些宏跟蹤調試
            ANSI標準說明了五個預定義的宏名。它們是:
            _LINE_ (兩個下劃線),對應%d
            _FILE_     對應%s
            _DATE_   對應%s
            _TIME_    對應%s
            _STDC_


            宏中"#"和"##"的用法
            我們使用#把宏參數變為一個字符串,用##把兩個宏參數貼合在一起.
            #define STR(s)     #s
            #define CONS(a,b) int(a##e##b)
            Printf(STR(vck));           // 輸出字符串"vck"
            printf("%d\n", CONS(2,3)); // 2e3 輸出:2000

            當宏參數是另一個宏的時候
            需要注意的是凡宏定義里有用"#"或"##"的地方宏參數是不會再展開.
            #define A          (2)
            #define STR(s)     #s
            #define CONS(a,b) int(a##e##b)
            printf("%s\n", CONS(A, A));               // compile error
            這一行則是:
            printf("%s\n", int(AeA));
            INT_MAX和A都不會再被展開, 然而解決這個問題的方法很簡單. 加多一層中間轉換宏.
            加這層宏的用意是把所有宏的參數在這層里全部展開, 那么在轉換宏里的那一個宏(_STR)就能得到正確的宏參數
            #define STR(s)      _STR(s)          // 轉換宏
            #define CONS(a,b)   _CONS(a,b)       // 轉換宏
            printf("int max: %s\n", STR(INT_MAX));          // INT_MAX,int型的最大值,為一個變量 #i nclude<climits>
            輸出為: int max: 0x7fffffff
            STR(INT_MAX) --> _STR(0x7fffffff) 然后再轉換成字符串;

            printf("%d\n", CONS(A, A));
            輸出為:200
            CONS(A, A) --> _CONS((2), (2)) --> int((2)e(2))

            "#"和"##"的一些應用特例
            1、合并匿名變量名
            #define ___ANONYMOUS1(type, var, line) type var##line
            #define __ANONYMOUS0(type, line) ___ANONYMOUS1(type, _anonymous, line)
            #define ANONYMOUS(type) __ANONYMOUS0(type, __LINE__)
            例:ANONYMOUS(static int); 即: static int _anonymous70; 70表示該行行號;
            第一層:ANONYMOUS(static int); --> __ANONYMOUS0(static int, __LINE__);
            第二層:                        --> ___ANONYMOUS1(static int, _anonymous, 70);
            第三層:                        --> static int _anonymous70;
            即每次只能解開當前層的宏,所以__LINE__在第二層才能被解開;

            注備:這里面的解釋 有點牽強。真正的解釋是這樣的:

               【宏參數的prescan】
               當一個宏參數被放進宏體時,這個宏參數會首先被全部展開(有例外,見下文)。當展開后的宏參數被放進宏體時,
               預處理器對新展開的宏體進行第二次掃描,并繼續展開。例如:
               #define PARAM( x ) x
               #define ADDPARAM( x ) INT_##x
               PARAM( ADDPARAM( 1 ) );
               因為ADDPARAM( 1 ) 是作為PARAM的宏參數,所以先將ADDPARAM( 1 )展開為INT_1,然后再將INT_1放進PARAM。
              
               例外情況是,如果PARAM宏里對宏參數使用了#或##,那么宏參數不會被展開:
               #define PARAM( x ) #x
               #define ADDPARAM( x ) INT_##x
               PARAM( ADDPARAM( 1 ) ); 將被展開為"ADDPARAM( 1 )"。

               使用這么一個規則,可以創建一個很有趣的技術:打印出一個宏被展開后的樣子,這樣可以方便你分析代碼:
               #define TO_STRING( x ) TO_STRING1( x )
               #define TO_STRING1( x ) #x
               TO_STRING首先會將x全部展開(如果x也是一個宏的話),然后再傳給TO_STRING1轉換為字符串,現在你可以這樣:
               const char *str = TO_STRING( PARAM( ADDPARAM( 1 ) ) );去一探PARAM展開后的樣子。

              預處理器的確是從內到外展開宏的

            2、填充結構
            #define FILL(a)   {a, #a}

            enum IDD{OPEN, CLOSE};
            typedef struct MSG{
            IDD id;
            const char * msg;
            }MSG;

            MSG _msg[] = {FILL(OPEN), FILL(CLOSE)};
            相當于:
            MSG _msg[] = {{OPEN, "OPEN"},
                          {CLOSE, "CLOSE"}};

            3、記錄文件名
            #define _GET_FILE_NAME(f)   #f
            #define GET_FILE_NAME(f)    _GET_FILE_NAME(f)
            static char FILE_NAME[] = GET_FILE_NAME(__FILE__);

            4、得到一個數值類型所對應的字符串緩沖大小
            #define _TYPE_BUF_SIZE(type) sizeof #type
            #define TYPE_BUF_SIZE(type)   _TYPE_BUF_SIZE(type)
            char buf[TYPE_BUF_SIZE(INT_MAX)];
                 --> char buf[_TYPE_BUF_SIZE(0x7fffffff)];
                 --> char buf[sizeof "0x7fffffff"];
            這里相當于:
            char buf[11];

            posted on 2012-11-02 17:40 Enic 閱讀(461) 評論(2)  編輯 收藏 引用 所屬分類: C/C++技巧

            評論

            # re: (轉)C宏技巧匯總 2012-11-26 14:37 xyl
            請問printf("%d\n", CONS(A, A));
            編譯通過了嗎?  回復  更多評論
              

            # re: (轉)C宏技巧匯總 2012-12-11 10:47 Enic
            你問的比較抽象喲@xyl
              回復  更多評論
              

            人妻少妇久久中文字幕一区二区 | 国产成人久久精品二区三区| 人妻精品久久久久中文字幕69 | 日韩va亚洲va欧美va久久| 久久久九九有精品国产| 日日狠狠久久偷偷色综合96蜜桃 | 欧美日韩中文字幕久久久不卡| 久久久久99精品成人片| 欧美久久久久久| 青青青国产精品国产精品久久久久| 精品久久人人妻人人做精品| 亚洲婷婷国产精品电影人久久| 精品无码久久久久久午夜| 久久精品国产免费一区| 亚洲伊人久久成综合人影院| 999久久久免费精品国产| 亚洲精品乱码久久久久久不卡| 亚洲国产成人久久精品影视| 日本WV一本一道久久香蕉| 国内精品伊人久久久久影院对白| 久久精品人人做人人爽电影蜜月| 久久99精品免费一区二区| 无码伊人66久久大杳蕉网站谷歌| 狠狠精品久久久无码中文字幕| 久久久久久夜精品精品免费啦| 久久精品国产亚洲av瑜伽| 精品久久久久久国产潘金莲| 精产国品久久一二三产区区别| 国产精品日韩欧美久久综合| 中文字幕乱码人妻无码久久| 区亚洲欧美一级久久精品亚洲精品成人网久久久久 | 久久久久人妻一区精品| 日韩人妻无码精品久久久不卡| 久久精品国产清自在天天线 | 亚洲精品白浆高清久久久久久| 人妻少妇精品久久| 99久久国产热无码精品免费| 久久国产乱子伦免费精品| 日日噜噜夜夜狠狠久久丁香五月| 天堂久久天堂AV色综合| 久久久久久综合网天天|