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

            久久人人爽人人爽人人片AV不| 久久精品免费观看| 国产精久久一区二区三区| 无码久久精品国产亚洲Av影片| 久久久久黑人强伦姧人妻| 国产ww久久久久久久久久| 国产精品久久久久jk制服| 91精品国产综合久久精品| 久久一日本道色综合久久| 久久精品中文字幕无码绿巨人| 亚洲国产另类久久久精品黑人| 伊人色综合九久久天天蜜桃| 亚洲国产精品综合久久一线| 欧洲性大片xxxxx久久久| 欧美国产成人久久精品| 人妻少妇久久中文字幕| 99久久久精品| 久久综合色之久久综合| 少妇无套内谢久久久久| 久久久久人妻精品一区| 久久婷婷久久一区二区三区| 亚洲午夜久久久精品影院 | 亚洲女久久久噜噜噜熟女| 国产激情久久久久久熟女老人| 国产麻豆精品久久一二三| 久久国产精品久久久| 久久久无码精品亚洲日韩软件| 一本色道久久综合狠狠躁篇| 国产成年无码久久久免费| www.久久热| 久久久高清免费视频| 久久777国产线看观看精品| 免费一级做a爰片久久毛片潮| 色8久久人人97超碰香蕉987| 91亚洲国产成人久久精品网址| 亚洲国产日韩欧美久久| 国产亚洲精品自在久久| 亚洲精品乱码久久久久久蜜桃| 粉嫩小泬无遮挡久久久久久| 亚洲国产精品狼友中文久久久| 99国产欧美精品久久久蜜芽|