• <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>

            大龍的博客

            常用鏈接

            統計

            最新評論

            用可變參數宏

            注意:_VA_ARGS__ 從VS2005才開始支持
            ===========
            在 GNU C 中,宏可以接受可變數目的參數,就象函數一樣,例如:
            #define pr_debug(fmt,arg...) printk(KERN_DEBUG fmt,##arg)
            用可變參數宏(variadic macros)傳遞可變參數表
            你可能很熟悉在函數中使用可變參數表,如:
            void printf(const char* format, …);
            直到最近,可變參數表還是只能應用在真正的函數中,不能使用在宏中。
            C99編譯器標準終于改變了這種局面,它允許你可以定義可變參數宏(variadic macros),這樣你就可以使用擁有可以變化的參數表的宏??勺儏岛昃拖裣旅孢@個樣子:
            #define debug(…) printf(__VA_ARGS__)
            缺省號代表一個可以變化的參數表。使用保留名 __VA_ARGS__ 把參數傳遞給宏。當宏的調用展開時,實際的參數就傳遞給 printf()了。例如:
            Debug(“Y = %d\n”, y);
            而處理器會把宏的調用替換成:
            printf(“Y = %d\n”, y);
            因為debug()是一個可變參數宏,你能在每一次調用中傳遞不同數目的參數:
            debug(“test”); //一個參數
            可變參數宏不被ANSI/ISO C++ 所正式支持。因此,你應當檢查你的編譯器,看它是否支持這項技術。

            用GCC和C99的可變參數宏, 更方便地打印調試信息
            gcc的預處理提供的可變參數宏定義真是好用:
            #ifdef DEBUG
             #define dbgprint(format,args...)  fprintf(stderr, format, ##args)
            #else
             #define dbgprint(format,args...)
            #endif
            如此定義之后,代碼中就可以用dbgprint了,例如dbgprint("aaa %s", __FILE__);。感覺這個功能比較Cool  
            下面是C99的方法:
            #define dgbmsg(fmt,...) printf(fmt,__VA_ARGS__)
             
            新的C99規范支持了可變參數的宏
            具體使用如下:
             以下內容為程序代碼:
             #include <stdarg.h> 
            #include <stdio.h>
             #define LOGSTRINGS(fm, ...) printf(fm,__VA_ARGS__)
             int main()
             {
                  LOGSTRINGS("hello, %d ", 10);
                  return 0;
             } 
             但現在似乎只有gcc才支持。
            可變參數的宏里的‘##’操作說明
            帶有可變參數的宏(Macros with a Variable Number of Arguments)
            在1999年版本的ISO C 標準中,宏可以象函數一樣,定義時可以帶有可變參數。宏的語法和函數的語法類似。下面有個例子:
            #define debug(format, ...) fprintf (stderr, format, __VA_ARGS__)
            這里,‘…’指可變參數。這類宏在被調用時,它(這里指‘…’)被表示成零個或多個符號,包括里面的逗號,一直到到右括弧結束為止。當被調用時,在宏體(macro body)中,那些符號序列集合將代替里面的__VA_ARGS__標識符。更多的信息可以參考CPP手冊。
            GCC始終支持復雜的宏,它使用一種不同的語法從而可以使你可以給可變參數一個名字,如同其它參數一樣。例如下面的例子:
            #define debug(format, args...) fprintf (stderr, format, args)
            這和上面舉的那個ISO C定義的宏例子是完全一樣的,但是這么寫可讀性更強并且更容易進行描述。
            GNU CPP還有兩種更復雜的宏擴展,支持上面兩種格式的定義格式。
            在標準C里,你不能省略可變參數,但是你卻可以給它傳遞一個空的參數。例如,下面的宏調用在ISO C里是非法的,因為字符串后面沒有逗號:
            debug ("A message")
            GNU CPP在這種情況下可以讓你完全的忽略可變參數。在上面的例子中,編譯器仍然會有問題(complain),因為宏展開后,里面的字符串后面會有個多余的逗號。
            為了解決這個問題,CPP使用一個特殊的‘##’操作。書寫格式為:
            #define debug(format, ...) fprintf (stderr, format, ## __VA_ARGS__)
            這里,如果可變參數被忽略或為空,‘##’操作將使預處理器(preprocessor)去除掉它前面的那個逗號。如果你在宏調用時,確實提供了一些可變參數,GNU CPP也會工作正常,它會把這些可變參數放到逗號的后面。象其它的pasted macro參數一樣,這些參數不是宏的擴展。

            怎樣寫參數個數可變的宏
            一種流行的技巧是用一個單獨的用括弧括起來的的 ``參數" 定義和調用宏, 參數在 宏擴展的時候成為類似 printf() 那樣的函數的整個參數列表。
                #define DEBUG(args) (printf("DEBUG: "), printf args)
                if(n != 0) DEBUG(("n is %d\n", n));
            明顯的缺陷是調用者必須記住使用一對額外的括弧。
            gcc 有一個擴展可以讓函數式的宏接受可變個數的參數。 但這不是標準。另一種 可能的解決方案是根據參數個數使用多個宏 (DEBUG1, DEBUG2, 等等), 或者用 逗號玩個這樣的花招:
                #define DEBUG(args) (printf("DEBUG: "), printf(args))
                #define _ ,
                DEBUG("i = %d" _ i);
            C99 引入了對參數個數可變的函數式宏的正式支持。在宏 ``原型" 的末尾加上符號 ... (就像在參數可變的函數定義中), 宏定義中的偽宏 __VA_ARGS__ 就會在調用是 替換成可變參數。
            最后, 你總是可以使用真實的函數, 接受明確定義的可變參數
            如果你需要替換宏, 使用一個 函數和一個非函數式宏, 如 #define printf myprintf。
             
             

            #include<stdio.h>
            #define DEBUG


            #ifdef DEBUG
            #define dprint(arg) printf arg
            #define dprint1(...) printf(__VA_ARGS__);
            #define dprint2(format,args...) fprintf(stderr,format,##args);
            #else
            #define dprint(arg) printf arg
            #define dprint1(...) 
            #define dprint2(format,args...) 
            #endif

            int main()
            {
                int y=100;
                double a=10;
                dprint(("y=%d a=%f\n",y,a));
                dprint1("y=%d a=%f\n",y,a);
                dprint2("y=%d a=%f\n",y,a);
                dprint2("hello world\n");
                
            }

            posted on 2010-12-03 16:49 大龍 閱讀(1598) 評論(0)  編輯 收藏 引用

            日韩人妻无码精品久久免费一| 久久久久亚洲av成人无码电影| 国产偷久久久精品专区| 亚洲AV无码久久| 久久99热国产这有精品| 久久久久女教师免费一区| 国产A级毛片久久久精品毛片| AAA级久久久精品无码片| 久久毛片免费看一区二区三区| 亚洲国产欧美国产综合久久| 国产国产成人精品久久| 久久精品一区二区三区AV| 久久亚洲国产欧洲精品一| 久久久高清免费视频| 久久香蕉国产线看观看99| 精品多毛少妇人妻AV免费久久| 久久精品国产福利国产秒| 亚洲色大成网站www久久九| 丰满少妇人妻久久久久久4| 久久丫精品国产亚洲av不卡| 久久中文精品无码中文字幕| 国产成人综合久久综合| 色欲av伊人久久大香线蕉影院| 亚洲精品国精品久久99热| 中文字幕亚洲综合久久| 久久精品人人槡人妻人人玩AV | 精品久久久久中文字幕日本| 久久国产精品免费一区| 秋霞久久国产精品电影院| 久久国产精品99精品国产| 久久狠狠爱亚洲综合影院| 亚洲国产精品无码久久九九| 99久久99久久精品国产片果冻 | 欧美日韩精品久久久久 | 久久久精品人妻一区二区三区四 | 亚洲午夜福利精品久久| 精品久久久久久国产牛牛app | 99久久国产亚洲高清观看2024| 嫩草影院久久99| 久久久青草青青亚洲国产免观| 久久国产精品99精品国产987|