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

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對不能為以下類型:
——char、signed char、unsigned char
——short、unsigned short
——signed short、short intsigned short int、unsigned 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語言繼承來的標準提升。
對于由省略號表示的參數,其實際參數在傳遞之前總執行這些提升(如果它們屬于需要提升的類型),將提升后的值傳遞給有關的函數?!g者注
                ——《C++程序設計語言》第3版-特別版 7.6 p138

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


這里有一個陷阱需要避免:
va_arg宏的第2個參數不能被指定為char、short或者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://m.shnenglu.com/ownwaterloo/archive/2009/04/21/is_va_end_necessary.html




Creative Commons License

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

轉載請注明 :
文章作者 - OwnWaterloo
發表時間 - 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 閱讀(13308) 評論(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年9月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

常用鏈接

留言簿(8)

隨筆檔案(16)

鏈接

搜索

  •  

積分與排名

  • 積分 - 198797
  • 排名 - 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>
            一区二区不卡在线视频 午夜欧美不卡在 | 久久亚洲高清| 正在播放欧美视频| 日韩天天综合| 亚洲视频一区| 亚洲免费视频中文字幕| 亚洲一区二区高清视频| 在线视频亚洲| 亚洲一区二区三区在线观看视频| 亚洲一区二区三区中文字幕在线| 国产精品99久久不卡二区| 亚洲影院免费观看| 久久狠狠亚洲综合| 欧美激情1区2区3区| 91久久精品一区| 国内精品嫩模av私拍在线观看| 韩国精品在线观看| 99精品热视频| 欧美在线啊v| 亚洲成人资源| 亚洲一区高清| 久久影视精品| 国产精品老牛| 亚洲国产成人av好男人在线观看| 一区二区三区国产精品| 久久黄金**| 日韩一级黄色片| 久久久噜噜噜久久中文字幕色伊伊| 欧美成人午夜免费视在线看片 | 一本一本a久久| 久久国产综合精品| 亚洲日本中文字幕免费在线不卡| 亚洲一区在线直播| 久久综合九九| 国产精品亚洲综合色区韩国| 亚洲国产一区二区三区a毛片| 亚洲欧美日韩爽爽影院| 欧美插天视频在线播放| 亚洲欧美精品伊人久久| 欧美福利电影网| 国产一区二区成人| 亚洲在线视频免费观看| 欧美成人激情在线| 午夜亚洲一区| 国产精品久久久亚洲一区| 亚洲日韩欧美视频一区| 久久嫩草精品久久久久| 宅男精品视频| 欧美日本韩国| 亚洲欧洲精品天堂一级 | 亚洲精品乱码久久久久久日本蜜臀| 一区二区福利| 久久综合一区| 欧美一区二区在线观看| 国产精品久久国产精品99gif| 亚洲精品久久久久| 欧美成人免费大片| 久久精品国产清自在天天线 | 久久精品一二三区| 国产一区二区三区四区hd| 亚洲欧美国产制服动漫| 99www免费人成精品| 欧美久久视频| 野花国产精品入口| 亚洲精品一区二区三区av| 模特精品裸拍一区| 亚洲精品国产拍免费91在线| 免费视频亚洲| 麻豆精品91| 亚洲精品小视频| 亚洲人午夜精品| 欧美另类极品videosbest最新版本| 在线国产精品一区| 欧美电影免费网站| 欧美国产亚洲另类动漫| 99re热精品| 中文日韩欧美| 国产午夜精品美女视频明星a级| 欧美专区日韩专区| 欧美自拍偷拍| 亚洲国产精品成人va在线观看| 欧美激情按摩在线| 欧美日韩情趣电影| 欧美在线不卡| 久久综合九色综合欧美就去吻| 亚洲激情第一页| 99视频一区| 国产欧美三级| 欧美91大片| 欧美日韩亚洲另类| 欧美中文字幕视频| 欧美超级免费视 在线| 一区二区三区久久久| 亚洲一区久久| 亚洲国产精品久久久久| 亚洲精品在线免费观看视频| 国产精品美女黄网| 另类av导航| 欧美日韩a区| 久久国产视频网站| 欧美黄色日本| 久久久久九九视频| 欧美日韩成人一区二区| 久久精品视频在线观看| 欧美激情区在线播放| 篠田优中文在线播放第一区| 久久久噜噜噜久久狠狠50岁| 亚洲午夜电影在线观看| 久久婷婷蜜乳一本欲蜜臀| 亚洲一区久久久| 久久中文字幕一区| 国产视频精品xxxx| 亚洲高清在线播放| 国产精品午夜在线观看| 欧美激情偷拍| 国产一区导航| 中文在线不卡视频| 亚洲人成高清| 久久成人18免费网站| 亚洲午夜国产成人av电影男同| 久久久久久婷| 久久成人免费电影| 欧美日韩免费观看一区三区| 欧美大片第1页| 国产自产2019最新不卡| 亚洲少妇一区| 亚洲美洲欧洲综合国产一区| 久久久久久亚洲精品杨幂换脸 | 欧美电影在线播放| 久久乐国产精品| 国产乱人伦精品一区二区 | 欧美v日韩v国产v| 久久久视频精品| 国产精品视频午夜| 中文av一区二区| 中日韩在线视频| 欧美精品尤物在线| 亚洲国产欧美在线人成| 在线观看视频欧美| 久久久人成影片一区二区三区观看 | 国产日韩欧美一区二区三区在线观看 | 午夜精品视频在线| 亚洲一区二区三区四区中文| 欧美日韩精品一区二区在线播放 | 一本色道久久加勒比精品| 欧美/亚洲一区| 欧美黄网免费在线观看| 亚洲伦伦在线| 欧美精品18| 亚洲肉体裸体xxxx137| 亚洲免费激情| 欧美午夜精品电影| 亚洲一区在线看| 欧美一级视频| 好吊妞**欧美| 免费在线欧美视频| 日韩一级大片在线| 久久av一区二区三区亚洲| 午夜精品国产精品大乳美女| 久久国产精品99国产| 国模精品一区二区三区色天香| 久久精品成人欧美大片古装| 欧美成人亚洲成人| 亚洲视频一二三| 国产一区二区高清| 毛片一区二区| 亚洲神马久久| 久久久久在线观看| 亚洲精品中文字| 国产精品入口尤物| 久久精品久久99精品久久| 欧美成人三级在线| 亚洲午夜视频在线| 极品尤物一区二区三区| 欧美日韩精品系列| 久久精品视频免费| 日韩视频免费观看| 久久琪琪电影院| 中文成人激情娱乐网| 精品成人乱色一区二区| 欧美精品在线免费| 久久九九有精品国产23| 日韩一级欧洲| 美女亚洲精品| 亚洲午夜精品福利| 亚洲第一久久影院| 国产精品久久久久av免费| 久久躁狠狠躁夜夜爽| 亚洲午夜国产成人av电影男同| 欧美gay视频激情| 欧美一级久久久| 亚洲视频香蕉人妖| 亚洲激情欧美| 国产在线视频欧美一区二区三区| 欧美久久久久久久久久| 久久午夜国产精品| 欧美一区二区日韩| 一区二区三区精品国产| 亚洲国产福利在线| 老牛国产精品一区的观看方式|