• <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
            實(shí)現(xiàn)一個(gè)有可變長(zhǎng)參數(shù)列表函數(shù)的時(shí)候,會(huì)使用到stdarg.h(這里不討論varargs.h)中提供的宏。

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


            這與《C++程序設(shè)計(jì)語(yǔ)言》中的一道練習(xí)題很類似。
            ——需要支持"%c"控制符

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





            簡(jiǎn)單的說(shuō),我們用va_arg(ap,type)取出一個(gè)參數(shù)的時(shí)候,
            type對(duì)不能為以下類型:
            ——char、signed char、unsigned char
            ——shortunsigned short
            ——signed short、short int、signed short intunsigned short int
            ——float


            一個(gè)簡(jiǎn)單的理由是:
            ——調(diào)用者絕對(duì)不會(huì)my_printf傳遞以上類型的實(shí)際參數(shù)。


            在C語(yǔ)言中,調(diào)用一個(gè)不帶原型聲明的函數(shù)時(shí):
            調(diào)用者會(huì)對(duì)每個(gè)參數(shù)執(zhí)行“默認(rèn)實(shí)際參數(shù)提升(default argument promotions)”。

            同時(shí),對(duì)可變長(zhǎng)參數(shù)列表超出最后一個(gè)類型聲明的形式參數(shù)之后的每一個(gè)實(shí)際參數(shù),也將執(zhí)行上述提升工作。
            提升工作如下:
            ——float類型的實(shí)際參數(shù)將提升到double
            ——char、short和相應(yīng)的signed、unsigned類型的實(shí)際參數(shù)提升到int
            ——如果int不能存儲(chǔ)原值,則提升到unsigned int

            然后,調(diào)用者將提升后的參數(shù)傳遞給被調(diào)用者。
            所以,my_printf是絕對(duì)無(wú)法接收到上述類型的實(shí)際參數(shù)的。




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

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

            這也是printf族函數(shù)沒(méi)有用于short和float的控制符的原因。



            附錄:

            在《C語(yǔ)言程序設(shè)計(jì)》對(duì)可變長(zhǎng)參數(shù)列表的相關(guān)章節(jié)中,并沒(méi)有提到這個(gè)陷阱。
            但是有提到默認(rèn)實(shí)際參數(shù)提升的規(guī)則:

            在沒(méi)有函數(shù)原型的情況下,char與short類型都將被轉(zhuǎn)換為int類型,float類型將被轉(zhuǎn)換為double類型。
                            ——《C語(yǔ)言程序設(shè)計(jì)》第2版  2.7 類型轉(zhuǎn)換 p36



            在其他一些書籍中,也有提到這個(gè)規(guī)則:


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

            …… float類型的參數(shù)會(huì)自動(dòng)轉(zhuǎn)換為double類型,short或char類型的參數(shù)會(huì)自動(dòng)轉(zhuǎn)換為int類型 ……
                            ——《C陷阱與缺陷》 4.4 形參、實(shí)參與返回值 p73


            這里有一個(gè)陷阱需要避免:
            va_arg宏的第2個(gè)參數(shù)不能被指定為char、short或者float類型。
            因?yàn)閏har和short類型的參數(shù)會(huì)被轉(zhuǎn)換為int類型,而float類型的參數(shù)會(huì)被轉(zhuǎn)換為double類型 ……
            例如,這樣寫肯定是不對(duì)的:
            c = va_arg(ap,char);
            因?yàn)槲覀儫o(wú)法傳遞一個(gè)char類型參數(shù),如果傳遞了,它將會(huì)被自動(dòng)轉(zhuǎn)化為int類型。上面的式子應(yīng)該寫成:
            c = va_arg(ap,int);
                            ——《C陷阱與缺陷》p164



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



            相關(guān)鏈接:

            ——《可變長(zhǎng)參數(shù)列表誤區(qū)與陷阱——va_end是必須的嗎?》
            http://m.shnenglu.com/ownwaterloo/archive/2009/04/21/is_va_end_necessary.html




            Creative Commons License

            作品采用知識(shí)共享署名-非商業(yè)性使用-相同方式共享 2.5 中國(guó)大陸許可協(xié)議進(jìn)行許可。

            轉(zhuǎn)載請(qǐng)注明 :
            文章作者 - OwnWaterloo
            發(fā)表時(shí)間 - 2009年04月21日
            原文鏈接 - http://m.shnenglu.com/ownwaterloo/archive/2009/04/21/unacceptable_type_in_va_arg.html

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

            FeedBack:
            # re: 可變長(zhǎng)參數(shù)列表誤區(qū)與陷阱——va_arg不可接受的類型
            2009-04-22 10:42 | vitacy
            va_arg(va_list,type)是自動(dòng)int對(duì)齊的。  回復(fù)  更多評(píng)論
              
            # re: 可變長(zhǎng)參數(shù)列表誤區(qū)與陷阱——va_arg不可接受的類型
            2009-04-22 15:32 | OwnWaterloo
            @vitacy

            1. C標(biāo)準(zhǔn)對(duì)默認(rèn)實(shí)際參數(shù)提升規(guī)則有明確規(guī)定。
            也就是說(shuō), 帶有可變長(zhǎng)參數(shù)列表的函數(shù), 絕對(duì)不會(huì)接受到char類型的實(shí)際參數(shù)。

            2. C標(biāo)準(zhǔn)對(duì)va_arg是否自動(dòng)對(duì)齊沒(méi)有任何說(shuō)明。
            你說(shuō)的va_arg(va_list,type)是自動(dòng)對(duì)齊, 只是在你的編譯器上。
            并不是所有編譯器都能自動(dòng)幫你完成這個(gè)工作。

            在所有C實(shí)現(xiàn)上, 能保證第1點(diǎn), 但不能保證第2點(diǎn)。
            依賴于第2點(diǎn), 代碼就依賴于特定編譯器。


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


            在gcc (GCC) 3.4.2 (mingw-special)中, type使用char, 會(huì)得到嚴(yán)重的警告:
            `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
            它會(huì)直接掛掉你的程序,來(lái)約束你必須按規(guī)矩辦事。
              回復(fù)  更多評(píng)論
              
            # re: 可變長(zhǎng)參數(shù)列表誤區(qū)與陷阱——va_arg不可接受的類型
            2009-12-14 16:23 | mikecheng
            你的說(shuō)法有問(wèn)題,估計(jì)你沒(méi)有仔細(xì)讀manual,在manual中就有用char的例子。  回復(fù)  更多評(píng)論
              
            # re: 可變長(zhǎng)參數(shù)列表誤區(qū)與陷阱——va_arg不可接受的類型
            2009-12-14 19:09 | OwnWaterloo
            @mikecheng
            你很悲劇,你仔細(xì)閱讀的manual中的那個(gè)例子是錯(cuò)的。
              回復(fù)  更多評(píng)論
              
            # re: 可變長(zhǎng)參數(shù)列表誤區(qū)與陷阱——va_arg不可接受的類型[未登錄](méi)
            2013-01-23 22:36 | alex
            干嗎要把float提升到double,就差沒(méi)有想到這一條,害我調(diào)試了半天。。float一般不是4個(gè)自己?jiǎn)??也算int對(duì)齊的呀,真奇怪。  回復(fù)  更多評(píng)論
              

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            <2025年6月>
            25262728293031
            1234567
            891011121314
            15161718192021
            22232425262728
            293012345

            常用鏈接

            留言簿(8)

            隨筆檔案(16)

            鏈接

            搜索

            •  

            積分與排名

            • 積分 - 198340
            • 排名 - 134

            最新隨筆

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            久久久久夜夜夜精品国产| 亚洲日本va中文字幕久久| 国产成人精品久久综合| 亚洲精品国产第一综合99久久| 欧美精品乱码99久久蜜桃| 久久久久久久人妻无码中文字幕爆 | 亚洲精品乱码久久久久久蜜桃不卡| 久久久久久久久久久| 97久久精品人妻人人搡人人玩| 国内精品伊人久久久久影院对白| 亚洲AV日韩AV永久无码久久| 99久久久久| 999久久久无码国产精品| 2021最新久久久视精品爱| 久久精品国产亚洲一区二区| 久久精品国产乱子伦| 久久久中文字幕日本| 久久久久国产精品| 麻豆一区二区99久久久久| 尹人香蕉久久99天天拍| 成人a毛片久久免费播放| 久久久无码一区二区三区| 99久久夜色精品国产网站| 久久久久人妻一区精品| 中文字幕久久欲求不满| 精品无码久久久久久尤物| 天堂久久天堂AV色综合| 7777精品伊人久久久大香线蕉| 日韩电影久久久被窝网| 久久精品国产亚洲Aⅴ蜜臀色欲| 国产精品美女久久久m| 久久99精品久久只有精品| 国内精品久久久久伊人av| 久久久久亚洲av无码专区喷水| 久久无码高潮喷水| 久久久精品国产免大香伊 | 久久亚洲私人国产精品| 日韩人妻无码一区二区三区久久 | 久久精品人人做人人妻人人玩 | 久久九色综合九色99伊人| 激情五月综合综合久久69|