• <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>
            posts - 16,  comments - 34,  trackbacks - 0
            實現一個有可變長參數列表函數的時候,會使用到stdarg.h(這里不討論varargs.h)中提供的宏。

            例如,我們要實現一個簡易的my_printf:
            1. 它只返回void, 不記錄輸出的字符數目
            2. 它只接受"%d"按整數輸出、"%c"按字符輸出、"%%"輸出'%'本身
            如下:
             1 #include <stdarg.h>
             2 
             3 void my_printf(const char* fmt, ... )
             4 {
             5     va_list ap;
             6     va_start(ap,fmt); /* 用最后一個具有參數的類型的參數去初始化ap */
             7     for (;*fmt;++fmt)
             8     {
             9         /* 如果不是控制字符 */
            10         if (*fmt!='%')
            11         {
            12             putchar(*fmt); /* 直接輸出 */
            13             continue;
            14         }
            15         /* 如果是控制字符,查看下一字符 */
            16         ++fmt;
            17         if ('\0'==*fmt) /* 如果是結束符 */
            18         {
            19             assert(0);  /* 這是一個錯誤 */
            20             break;
            21         }
            22         switch (*fmt)
            23         {
            24         case '%'/* 連續2個'%'輸出1個'%' */
            25             putchar('%');
            26             break;
            27         case 'd'/* 按照int輸出 */
            28             {
            29                 /* 下一個參數是int,取出 */
            30                 int i = va_arg(ap,int);
            31                 printf("%d",i);
            32             }
            33             break;
            34         case 'c'/* 按照字符輸出 */
            35             {
            36                 /*但是,下一個參數是char嗎*/
            37                 /*  可以這樣取出嗎? */
            38                 char c = va_arg(ap,char);
            39                 printf("%c",c);
            40             }
            41             break;
            43         }
            44     }
            45     va_end(ap);  /* 釋放ap—— 必須! 見相關鏈接*/
            46 }


            這與《C++程序設計語言》中的一道練習題很類似。
            ——需要支持"%c"控制符

            在《C++程序設計語言-題解》中,給出了一個答案(中文p65頁)。
            但是, 如同上面的代碼一樣,它們都是錯誤的!





            簡單的說,我們用va_arg(ap,type)取出一個參數的時候,
            type對不能為以下類型:
            ——charsigned charunsigned char
            ——shortunsigned short
            ——signed shortshort intsigned short intunsigned short int
            ——float


            一個簡單的理由是:
            ——調用者絕對不my_printf傳遞以上類型的實際參數


            在C語言中,調用一個不帶原型聲明的函數時:
            調用者會對每個參數執行“默認實際參數提升(default argument promotions)”。

            同時,對可變長參數列表超出最后一個類型聲明的形式參數之后的每一個實際參數,也將執行上述提升工作。
            提升工作如下:
            ——float類型的實際參數將提升到double
            ——char、short和相應的signed、unsigned類型的實際參數提升到int
            ——如果int不能存儲原值,則提升到unsigned int

            然后,調用者將提升后的參數傳遞給被調用者。
            所以,my_printf是絕對無法接收到上述類型的實際參數的。




            上面的代碼的38與39行,應該改為:
            int c = va_arg(ap,int);
            printf(
            "%c",c);

            同理, 如果需要使用short和float, 也應該這樣:
            short s = (short)va_arg(ap,int);
            float f = (float)va_arg(ap,double);

            這也是printf族函數沒有用于short和float的控制符的原因。



            附錄:

            在《C語言程序設計》對可變長參數列表的相關章節中,并沒有提到這個陷阱。
            但是有提到默認實際參數提升的規則:

            在沒有函數原型的情況下,char與short類型都將被轉換為int類型,float類型將被轉換為double類型。
                            ——《C語言程序設計》第2版  2.7 類型轉換 p36



            在其他一些書籍中,也有提到這個規則:


            事情很清楚,如果一個參數沒有聲明,編譯器就沒有信息去對它執行標準的類型檢查和轉換。
            在這種情況下,一個char或short將作為int傳遞,float將作為double傳遞。
            這些做未必是程序員所期望的。
            腳注:這些都是由C語言繼承來的標準提升。
            對于由省略號表示的參數,其實際參數在傳遞之前總執行這些提升(如果它們屬于需要提升的類型),將提升后的值傳遞給有關的函數。——譯者注
                            ——《C++程序設計語言》第3版-特別版 7.6 p138

            …… float類型的參數會自動轉換為double類型,short或char類型的參數會自動轉換為int類型 ……
                            ——《C陷阱與缺陷》 4.4 形參、實參與返回值 p73


            這里有一個陷阱需要避免:
            va_arg宏的第2個參數不能被指定為charshort或者float類型。
            因為char和short類型的參數會被轉換為int類型,而float類型的參數會被轉換為double類型 ……
            例如,這樣寫肯定是不對的:
            c = va_arg(ap,char);
            因為我們無法傳遞一個char類型參數,如果傳遞了,它將會被自動轉化為int類型。上面的式子應該寫成:
            c = va_arg(ap,int);
                            ——《C陷阱與缺陷》p164



            2009/05/07 修改:
            printf函數族有用于short的控制符“h”。
            見:http://www.cplusplus.com/reference/clibrary/cstdio/printf/



            相關鏈接:

            ——《可變長參數列表誤區與陷阱——va_end是必須的嗎?》
            http://www.shnenglu.com/ownwaterloo/archive/2009/04/21/is_va_end_necessary.html




            Creative Commons License

            作品采用知識共享署名-非商業性使用-相同方式共享 2.5 中國大陸許可協議進行許可。

            轉載請注明 :
            文章作者 - OwnWaterloo
            發表時間 - 2009年04月21日
            原文鏈接 - http://www.shnenglu.com/ownwaterloo/archive/2009/04/21/unacceptable_type_in_va_arg.html

            posted on 2009-04-21 23:41 OwnWaterloo 閱讀(13217) 評論(5)  編輯 收藏 引用

            FeedBack:
            # re: 可變長參數列表誤區與陷阱——va_arg不可接受的類型
            2009-04-22 10:42 | vitacy
            va_arg(va_list,type)是自動int對齊的。  回復  更多評論
              
            # re: 可變長參數列表誤區與陷阱——va_arg不可接受的類型
            2009-04-22 15:32 | OwnWaterloo
            @vitacy

            1. C標準對默認實際參數提升規則有明確定。
            也就是說, 帶有可變長參數列表的函數, 絕對不會接受到char類型的實際參數。

            2. C標準對va_arg是否自動對齊沒有任何說明
            你說的va_arg(va_list,type)是自動對齊, 只是在你的編譯器上。
            并不是所有編譯器都能自動幫你完成這個工作。

            在所有C實現上, 能保證第1點, 但不能保證第2點。
            依賴于第2點, 代碼就依賴于特定編譯器。


            你說va_arg(ap,type)是自動對齊, 證明你有研究過。
            喜歡作這些研究的, 都是聰明的家伙。
            但聰明的家伙總喜歡不按規矩辦事


            在gcc (GCC) 3.4.2 (mingw-special)中, type使用char, 會得到嚴重的警告:
            `char' is promoted to `int' when passed through `...'
            (so you should pass `int' not `char' to `va_arg')
            note: if this code is reached, the program will abort
            它會直接掛掉你的程序,來約束你必須按規矩辦事。
              回復  更多評論
              
            # re: 可變長參數列表誤區與陷阱——va_arg不可接受的類型
            2009-12-14 16:23 | mikecheng
            你的說法有問題,估計你沒有仔細讀manual,在manual中就有用char的例子。  回復  更多評論
              
            # re: 可變長參數列表誤區與陷阱——va_arg不可接受的類型
            2009-12-14 19:09 | OwnWaterloo
            @mikecheng
            你很悲劇,你仔細閱讀的manual中的那個例子是錯的。
              回復  更多評論
              
            # re: 可變長參數列表誤區與陷阱——va_arg不可接受的類型[未登錄]
            2013-01-23 22:36 | alex
            干嗎要把float提升到double,就差沒有想到這一條,害我調試了半天。。float一般不是4個自己嗎?也算int對齊的呀,真奇怪。  回復  更多評論
              
            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            常用鏈接

            留言簿(8)

            隨筆檔案(16)

            鏈接

            搜索

            •  

            積分與排名

            • 積分 - 197903
            • 排名 - 133

            最新隨筆

            最新評論

            閱讀排行榜

            評論排行榜

            天天综合久久一二三区| 91性高湖久久久久| 久久精品一区二区| 亚洲国产成人精品91久久久| 亚洲欧美伊人久久综合一区二区| 亚洲综合久久综合激情久久 | 亚洲精品无码久久久久| 国产精品视频久久久| 无码人妻少妇久久中文字幕| 久久综合给合久久狠狠狠97色| 日韩精品久久久久久| 久久久高清免费视频| 日韩欧美亚洲综合久久影院d3| 一个色综合久久| 久久精品无码一区二区app| 日韩精品久久久久久免费| 午夜视频久久久久一区 | 国产V综合V亚洲欧美久久| 午夜精品久久久内射近拍高清| 久久99国产精品久久久| 人妻精品久久久久中文字幕69| 久久久久亚洲AV无码专区网站 | 91精品国产91久久久久福利| 久久久久久精品无码人妻| 久久国产视屏| 国产精品VIDEOSSEX久久发布| 精品久久久无码人妻中文字幕豆芽| 模特私拍国产精品久久| 久久婷婷五月综合成人D啪| 国产福利电影一区二区三区久久老子无码午夜伦不 | 久久婷婷五月综合成人D啪 | 亚洲精品午夜国产VA久久成人 | 国产成人99久久亚洲综合精品| 久久99热国产这有精品| 精品久久人妻av中文字幕| 久久超乳爆乳中文字幕| 久久婷婷五月综合色奶水99啪| 久久人人添人人爽添人人片牛牛| 久久高潮一级毛片免费| 久久久久久A亚洲欧洲AV冫| 久久精品国产一区二区电影|