• <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
              回復  更多評論
              

            少妇久久久久久久久久| 久久亚洲色一区二区三区| 久久SE精品一区二区| 亚洲欧美伊人久久综合一区二区 | 久久久久亚洲av成人网人人软件| 久久久国产99久久国产一| 亚洲精品tv久久久久| 久久99精品久久久久久久久久| 久久99久久99小草精品免视看| 欧美日韩成人精品久久久免费看 | 国产午夜久久影院| 午夜精品久久久久久影视riav| 久久久久人妻精品一区二区三区 | 久久亚洲欧美日本精品| 无码国内精品久久综合88| 久久精品国产亚洲一区二区| 亚洲精品第一综合99久久| 狠狠色婷婷综合天天久久丁香| 久久精品综合网| 久久久网中文字幕| 亚洲伊人久久大香线蕉苏妲己| 亚洲国产精品无码久久久蜜芽| 久久99精品久久久久久9蜜桃| 色婷婷综合久久久中文字幕| 一本久久免费视频| 久久久久久青草大香综合精品| 久久精品aⅴ无码中文字字幕不卡| 日本WV一本一道久久香蕉| 99久久99久久精品国产片果冻| 狠狠色丁香久久综合婷婷| 久久精品麻豆日日躁夜夜躁| 伊人久久无码中文字幕| 老男人久久青草av高清| 天天影视色香欲综合久久| 热久久国产欧美一区二区精品| a级毛片无码兔费真人久久| 91精品国产色综久久| 国产精品免费看久久久香蕉| 狠狠色丁香婷婷综合久久来来去 | 精品久久久无码中文字幕| 91精品无码久久久久久五月天|