青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

posts - 16,  comments - 34,  trackbacks - 0
實(shí)現(xiàn)一個(gè)有可變長參數(shù)列表函數(shù)的時(shí)候,會使用到stdarg.h(這里不討論varargs.h)中提供的宏。

例如,我們要實(shí)現(xiàn)一個(gè)簡易的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ì)語言》中的一道練習(xí)題很類似。
——需要支持"%c"控制符

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





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


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


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

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

然后,調(diào)用者將提升后的參數(shù)傳遞給被調(diào)用者。
所以,my_printf是絕對無法接收到上述類型的實(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ù)沒有用于short和float的控制符的原因。



附錄:

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

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



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


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

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


這里有一個(gè)陷阱需要避免:
va_arg宏的第2個(gè)參數(shù)不能被指定為charshort或者float類型。
因?yàn)閏har和short類型的參數(shù)會被轉(zhuǎn)換為int類型,而float類型的參數(shù)會被轉(zhuǎn)換為double類型 ……
例如,這樣寫肯定是不對的:
c = va_arg(ap,char);
因?yàn)槲覀儫o法傳遞一個(gè)char類型參數(shù),如果傳遞了,它將會被自動轉(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)鏈接:

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




Creative Commons License

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

轉(zhuǎn)載請注明 :
文章作者 - OwnWaterloo
發(fā)表時(shí)間 - 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 閱讀(13306) 評論(5)  編輯 收藏 引用

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

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

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

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


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


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

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


<2025年9月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

常用鏈接

留言簿(8)

隨筆檔案(16)

鏈接

搜索

  •  

積分與排名

  • 積分 - 198763
  • 排名 - 134

最新隨筆

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            一本一本久久a久久精品综合麻豆 一本一本久久a久久精品牛牛影视 | 一本大道久久精品懂色aⅴ| 老司机午夜精品视频在线观看| 国内外成人免费激情在线视频| 久久久久久久精| 久久久久久91香蕉国产| 亚洲大片免费看| 欧美精品在线一区二区| 国产九九精品视频| 久久精品免费播放| 蜜臀av一级做a爰片久久| 一本久久综合亚洲鲁鲁| 亚洲字幕一区二区| 伊人春色精品| 99riav国产精品| 国产一区二区你懂的| 亚洲第一精品夜夜躁人人爽| 麻豆精品视频在线| 亚洲在线第一页| 久久精品国产免费看久久精品| 亚洲国产精品第一区二区| 亚洲精品在线视频| 精品动漫一区| 欧美一级片久久久久久久| 久久精品首页| 一区二区欧美日韩| 欧美在线免费一级片| 一区二区三区黄色| 久久久999精品视频| 亚洲视频久久| 老司机成人网| 久久国产欧美| 欧美三级韩国三级日本三斤| 久久综合五月| 国产精品久久久久久久久免费桃花| 麻豆精品网站| 亚洲五月六月| 亚洲精品五月天| 国产精品久久久久av免费| 国产精品99一区| 欧美国产成人精品| 国产免费成人av| 99re亚洲国产精品| 亚洲国产高清视频| 亚洲无线一线二线三线区别av| 久久综合色8888| 国产精品一区二区在线| 亚洲麻豆av| 亚洲精品国产精品乱码不99 | 久久精品国产精品亚洲综合| 欧美视频四区| 亚洲经典在线看| 亚洲国产高清在线| 久久天天躁夜夜躁狠狠躁2022| 性欧美在线看片a免费观看| 欧美精品一区二区视频 | 国产欧美日韩视频在线观看 | 亚洲精品综合久久中文字幕| 亚洲二区在线| 久久这里只有| 欧美国产先锋| 在线亚洲激情| 国产精品最新自拍| 羞羞色国产精品| 久久国产黑丝| 亚洲综合首页| 亚洲国产电影| 久久乐国产精品| 欧美黄污视频| 99re亚洲国产精品| 欧美日韩免费观看一区=区三区| 亚洲国产一区二区a毛片| 亚洲精品在线免费观看视频| 欧美精品成人在线| 在线视频欧美日韩精品| 午夜天堂精品久久久久| 国产精品视频99| 欧美在现视频| 亚洲国产精品久久久| 亚洲美女在线看| 亚洲国产高清aⅴ视频| 99视频一区| 国产精品久久久久久五月尺| 欧美一区=区| 欧美高清视频一区| 亚洲图片欧洲图片av| 欧美国产日韩一区| 国产精品亚洲综合| 久久久久一区二区三区四区| 亚洲国产精品激情在线观看| 亚洲图片欧美一区| 黑人一区二区| 欧美日本亚洲| 久久av一区二区| 亚洲激情第一区| 欧美亚洲综合另类| 亚洲精品在线观看免费| 国产精品一区二区男女羞羞无遮挡 | 亚洲欧美中日韩| 一区二区在线观看视频| 欧美激情四色 | 亚洲美女电影在线| 国产精品一区二区三区免费观看 | 中日韩美女免费视频网站在线观看| 久久精品国产精品 | 久久精品国产99国产精品| 亚洲欧洲日本一区二区三区| 国产视频久久| 欧美日韩久久不卡| 久久久夜夜夜| 午夜精品久久久| 最新日韩在线视频| 久久午夜精品| 午夜视频在线观看一区二区三区| 亚洲国产欧美一区二区三区久久 | 亚洲欧美日本国产专区一区| 亚洲国产第一| 激情久久五月| 国产精品日韩精品欧美精品| 亚洲国产成人精品久久| 久久夜色精品亚洲噜噜国产mv | 99国产精品久久久久久久成人热 | 国产欧美日韩一区二区三区在线| 91久久精品久久国产性色也91 | 欧美午夜剧场| 免费看精品久久片| 久久精品国产77777蜜臀| 亚洲欧美日韩在线综合| 在线亚洲伦理| 99在线热播精品免费99热| 亚洲国产欧洲综合997久久| 久久国产精品亚洲77777| 亚洲欧美另类综合偷拍| 亚洲视频免费在线观看| 9色porny自拍视频一区二区| 亚洲欧洲一区二区三区| 亚洲国产高清在线观看视频| 一区视频在线播放| 又紧又大又爽精品一区二区| 国产婷婷精品| 欧美综合国产精品久久丁香| 欧美亚洲免费高清在线观看| 亚洲一区二区四区| 亚洲欧美激情一区| 欧美一级久久久久久久大片| 午夜一区不卡| 久久九九99视频| 麻豆成人精品| 欧美日韩123| 国产精品久久国产愉拍| 一区二区三区视频在线观看| 亚洲网在线观看| 亚洲欧美激情视频| 久久国产日韩欧美| 久久这里只精品最新地址| 免费欧美在线视频| 亚洲黄色天堂| 在线综合亚洲| 欧美一区二区三区在线观看 | 亚洲精品人人| 亚洲在线观看免费| 久久久久91| 欧美精品久久久久久久久老牛影院 | 欧美freesex8一10精品| 亚洲国产精品视频| 99re8这里有精品热视频免费 | 欧美在线视频网站| 狂野欧美性猛交xxxx巴西| 亚洲电影av在线| 亚洲午夜精品网| 久久久精品国产免大香伊| 免费亚洲一区二区| 新狼窝色av性久久久久久| 亚洲精品国产精品国自产在线| 亚洲制服av| 欧美 亚欧 日韩视频在线| 日韩一二三在线视频播| 久久精品国产精品亚洲精品| 欧美精品一区二区三区蜜臀| 国产精品一区二区三区观看| 亚洲国产精品久久久久婷婷884 | 亚洲第一精品在线| 亚洲欧美高清| 欧美精品电影| 国内精品久久久久影院优| 99精品国产在热久久| 久久精品国产2020观看福利| 91久久精品美女高潮| 午夜精品久久久久久久白皮肤| 亚洲国产日韩美| 亚洲砖区区免费| 国产精品日韩电影| 91久久精品国产91久久性色| 香蕉久久夜色精品| 亚洲清纯自拍| 久久人人爽爽爽人久久久| 国产精品久久久久久久久动漫| 一区二区三区鲁丝不卡| 欧美在线视频一区| 极品中文字幕一区|