• <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++博客 首頁 新隨筆 聯(lián)系 聚合 管理
              217 Posts :: 61 Stories :: 32 Comments :: 0 Trackbacks

            C 宏

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


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

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

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

            "#"和"##"的一些應(yīng)用特例
            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表示該行行號(hào);
            第一層:ANONYMOUS(static int); --> __ANONYMOUS0(static int, __LINE__);
            第二層:                        --> ___ANONYMOUS1(static int, _anonymous, 70);
            第三層:                        --> static int _anonymous70;
            即每次只能解開當(dāng)前層的宏,所以__LINE__在第二層才能被解開;

            注備:這里面的解釋 有點(diǎn)牽強(qiáng)。真正的解釋是這樣的:

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

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

              預(yù)處理器的確是從內(nèi)到外展開宏的

            2、填充結(jié)構(gòu)
            #define FILL(a)   {a, #a}

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

            MSG _msg[] = {FILL(OPEN), FILL(CLOSE)};
            相當(dāng)于:
            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、得到一個(gè)數(shù)值類型所對應(yīng)的字符串緩沖大小
            #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"];
            這里相當(dāng)于:
            char buf[11];

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

            評論

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

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

            亚洲国产成人精品女人久久久 | yy6080久久| 亚洲国产成人精品91久久久| 久久综合丁香激情久久| 国产成人久久精品区一区二区| 日产精品久久久久久久性色| 亚洲国产精品无码久久久秋霞2| 精品伊人久久久| 国产亚洲精久久久久久无码77777| 亚洲精品综合久久| 伊人久久大香线蕉AV色婷婷色| 亚洲精品乱码久久久久久蜜桃不卡| 久久精品国产亚洲αv忘忧草| 国内高清久久久久久| 99久久成人国产精品免费| 久久线看观看精品香蕉国产| 国内精品久久久久久久亚洲| 久久久久九国产精品| 女人高潮久久久叫人喷水| 亚洲色大成网站www久久九| 成人免费网站久久久| 久久久久国产日韩精品网站| 久久无码AV一区二区三区| 亚洲午夜久久久久久久久久| 久久精品免费一区二区三区| 久久久久无码国产精品不卡| 7777精品久久久大香线蕉| 91精品国产综合久久久久久| 久久久久亚洲AV综合波多野结衣| 久久亚洲AV成人无码| 久久精品国产免费| 99久久国产宗和精品1上映| 日本精品久久久中文字幕 | 国产精品免费久久久久影院| 香蕉aa三级久久毛片| 久久精品麻豆日日躁夜夜躁| 久久国产精品二国产精品| 18岁日韩内射颜射午夜久久成人 | 无码人妻久久一区二区三区| 久久免费国产精品一区二区| 久久精品国产精品亚洲精品|