• <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++博客 首頁(yè) 新隨筆 聯(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)換為大寫(xiě)
            #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)說(shuō)明了五個(gè)預(yù)定義的宏名。它們是:
            _LINE_ (兩個(gè)下劃線),對(duì)應(yīng)%d
            _FILE_     對(duì)應(yīng)%s
            _DATE_   對(duì)應(yīng)%s
            _TIME_    對(duì)應(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ì)再展開(kāi).
            #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ì)再被展開(kāi), 然而解決這個(gè)問(wèn)題的方法很簡(jiǎn)單. 加多一層中間轉(zhuǎn)換宏.
            加這層宏的用意是把所有宏的參數(shù)在這層里全部展開(kāi), 那么在轉(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;
            即每次只能解開(kāi)當(dāng)前層的宏,所以__LINE__在第二層才能被解開(kāi);

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

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

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

              預(yù)處理器的確是從內(nèi)到外展開(kā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ù)值類(lèi)型所對(duì)應(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 閱讀(472) 評(píng)論(2)  編輯 收藏 引用 所屬分類(lèi): C/C++技巧

            評(píng)論

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

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

            久久久久久综合网天天| 久久av免费天堂小草播放| 伊人久久大香线蕉成人| 99精品国产免费久久久久久下载| 久久久久亚洲国产| 国产成人久久精品激情| 99久久夜色精品国产网站| 一97日本道伊人久久综合影院| 久久久久久综合网天天| 很黄很污的网站久久mimi色| 精品久久亚洲中文无码| 99久久亚洲综合精品网站| 久久午夜无码鲁丝片秋霞| 精品久久香蕉国产线看观看亚洲| 久久天天日天天操综合伊人av| 亚洲国产美女精品久久久久∴| 久久国产V一级毛多内射| 一本色综合网久久| 亚洲国产成人久久综合碰| 久久综合丝袜日本网| 人妻少妇久久中文字幕| 久久久精品日本一区二区三区| 精品久久久久久成人AV| 亚洲精品乱码久久久久久不卡| 99久久精品免费看国产一区二区三区| 久久亚洲AV无码精品色午夜| 久久国产精品免费一区二区三区| av无码久久久久久不卡网站| 欧美一区二区三区久久综合| 久久久久久久久波多野高潮| 欧美日韩精品久久久久| 久久99国产精品成人欧美| 91精品国产91久久久久久蜜臀 | 久久国产免费观看精品| 四虎国产精品成人免费久久| 色综合久久天天综线观看| 国产呻吟久久久久久久92| 精品无码久久久久久久动漫| 99久久精品免费看国产| 久久久久国产精品麻豆AR影院| 久久99精品久久久久久野外|