| 導購 | 订阅 | 在线投稿
分享
 
 
當前位置: 王朝網路 >> c/c++ >> 水滴石穿C語言之可變參數問題
 

水滴石穿C語言之可變參數問題

2008-06-01 02:07:24  編輯來源:互聯網  简体版  手機版  評論  字體: ||
 
 
  C語言中有一種長度不確定的參數,形如:"…",它主要用在參數個數不確定的函數中,我們最輕易想到的例子是PRintf函數。

  原型:

  int printf( const char *format [, argument]... );

  使用例:printf("Enjoy yourself everyday!\n");

  printf("The value is %d!\n", value);

  這種可變參數可以說是C語言一個比較難理解的部分,這裏會由幾個問題引發一些對它的分析。

  注重:在C++中有函數重載(overload)可以用來區別不同函數參數的調用,但它還是不能表示任意數量的函數參數。

  問題:printf的實現

  請問,如何自己實現printf函數,如何處理其中的可變參數問題? 答案與分析:

  在標准C語言中定義了一個頭文件專門用來對付可變參數列表,它包含了一組宏,和一個va_list的typedef聲明。一個典型實現如下:

  typedef char* va_list;

  #define va_start(list) list = (char*)&va_alist

  #define va_end(list)

  #define va_arg(list, mode)\

  ((mode*) (list += sizeof(mode)))[-1]

  自己實現printf:

  #include

  int printf(char* format, …)

  {

  va_list ap;

  va_start(ap, format);

  int n = vprintf(format, ap);

  va_end(ap);

  return n;

  }

  問題:運行時才確定的參數

  有沒有辦法寫一個函數,這個函數參數的具體形式可以在運行時才確定?

  答案與分析:

  目前沒有"正規"的解決辦法,不過獨門偏方倒是有一個,因爲有一個函數已經給我們做出了這方面的榜樣,那就是main(),它的原型是:

  int main(int argc,char *argv[]);

  函數的參數是argc和argv。

  深入想一下,"只能在運行時確定參數形式",也就是說你沒辦法從聲明中看到所接受的參數,也即是參數根本就沒有固定的形式。 常用的辦法是你可以通過定義一個void *類型的參數,用它來指向實際的參數區,然後在函數中根據根據需要任意解釋它們的含義。這就是main函數中argv的含義,而argc,則用來表明實際的參數個數,這爲我們使用提供了進一步的方便,當然,這個參數不是必需的。

  雖然參數沒有固定形式,但我們必然要在函數中解析參數的意義,因此,理所當然會有一個要求,就是調用者和被調者之間要對參數區內容的格式,大小,有效性等所有方面達成一致,否則南轅北轍各說各話就慘了。

  問題:可變長參數的傳遞

  有時候,需要編寫一個函數,將它的可變長參數直接傳遞給另外的函數,請問,這個要求能否實現?

  答案與分析:

  目前,你尚無辦法直接做到這一點,但是我們可以迂回前進,首先,我們定義被調用函數的參數爲va_list類型,同時在調用函數中將可變長參數列表轉換爲va_list,這樣就可以進行變長參數的傳遞了。看如下所示:

  void subfunc (char *fmt, va_list argp)

  {

  ...

  arg = va_arg (fmt, argp); /* 從argp中逐一取出所要的參數 */

  ...

  }

  void mainfunc (char *fmt, ...)

  {

  va_list argp;

  va_start (argp, fmt); /* 將可變長參數轉換爲va_list */

  subfunc (fmt, argp); /* 將va_list傳遞給子函數 */

  va_end (argp);

  ...

  }

  問題:可變長參數中類型爲函數指針

  我想使用va_arg來提取出可變長參數中類型爲函數指針的參數,結果卻總是不正確,爲什麽?

  答案與分析:

  這個與va_arg的實現有關。一個簡單的、演示版的va_arg實現如下:

  #define va_arg(argp, type) \

  (*(type *)(((argp) += sizeof(type)) - sizeof(type)))

  其中,argp的類型是char *。

  假如你想用va_arg從可變參數列表中提取出函數指針類型的參數,例如

  int (*)(),則va_arg(argp, int (*)())被擴展爲:

  (*(int (*)() *)(((argp) += sizeof (int (*)())) -sizeof (int (*)())))

  顯然,(int (*)() *)是無意義的。

  解決這個問題的辦法是將函數指針用typedef定義成一個獨立的數據類型,例如:

  typedef int (*funcptr)();

  這時候再調用va_arg(argp, funcptr)將被擴展爲:

  (* (funcptr *)(((argp) += sizeof (funcptr)) - sizeof (funcptr)))

  這樣就可以通過編譯檢查了。

  問題:可變長參數的獲取

  有這樣一個具有可變長參數的函數,其中有下列代碼用來獲取類型爲float的實參:

  va_arg (argp, float);

  這樣做可以嗎?

  答案與分析:

  不可以。在可變長參數中,應用的是"加寬"原則。也就是float類型被擴展成double;char, short被擴展成int。因此,假如你要去可變長參數列表中原來爲float類型的參數,需要用va_arg(argp, double)。對char和short類型的則用va_arg(argp, int)。

  

  

  問題:定義可變長參數的一個限制

  爲什麽我的編譯器不答應我定義如下的函數,也就是可變長參數,但是沒有任何的固定參數?

  int f (...)

  {

  ...

  }

  答案與分析:

  不可以。這是ANSI C 所要求的,你至少得定義一個固定參數。

  這個參數將被傳遞給va_start(),然後用va_arg()和va_end()來確定所有實際調用時可變長參數的類型和值。
水滴石穿C語言之可變參數問題
更多內容請看C/C++進階技術文檔專題,或
 
 
 
上一篇《數據結構學習(C++)之圖》
下一篇《高質量C++/C編程指南(七)》
 
 
 
 
 
 
日版寵物情人插曲《Winding Road》歌詞

日版寵物情人2017的插曲,很帶節奏感,日語的,女生唱的。 最後聽見是在第8集的時候女主手割傷了,然後男主用嘴幫她吸了一下,插曲就出來了。 歌手:Def...

兄弟共妻,我成了他們夜裏的美食

老鍾家的兩個兒子很特別,就是跟其他的人不太一樣,魔一般的執著。兄弟倆都到了要結婚的年齡了,不管自家老爹怎麽磨破嘴皮子,兄弟倆說不娶就不娶,老父母爲兄弟兩操碎了心...

如何磨出破洞牛仔褲?牛仔褲怎麽剪破洞?

把牛仔褲磨出有線的破洞 1、具體工具就是磨腳石,下面墊一個硬物,然後用磨腳石一直磨一直磨,到把那塊磨薄了,用手撕開就好了。出來的洞啊很自然的。需要貓須的話調幾...

我就是掃描下圖得到了敬業福和愛國福

先來看下敬業福和愛國福 今年春節,支付寶再次推出了“五福紅包”活動,表示要“把欠大家的敬業福都還給大家”。 今天該活動正式啓動,和去年一樣,需要收集“五福”...

冰箱異味産生的原因和臭味去除的方法

有時候我們打開冰箱就會聞到一股異味,冰箱裏的這種異味是因爲一些物質發出的氣味的混合體,聞起來讓人惡心。 産生這些異味的主要原因有以下幾點。 1、很多人有這種習...

《極品家丁》1-31集大結局分集劇情介紹

簡介 《極品家丁》講述了現代白領林晚榮無意回到古代金陵,並追隨蕭二小姐化名“林三”進入蕭府,不料卻陰差陽錯上演了一出低級家丁拼搏上位的“林三升職記”。...

李溪芮《極品家丁》片尾曲《你就是我最愛的寶寶》歌詞

你就是我最愛的寶寶 - 李溪芮 (電視劇《極品家丁》片尾曲) 作詞:常馨內 作曲:常馨內 你的眉 又鬼馬的挑 你的嘴 又壞壞的笑 上一秒吵鬧 下...

烏梅的功效與作用以及烏梅的食用禁忌有哪些?

烏梅,又稱春梅,中醫認爲,烏梅味酸,性溫,無毒,具有安心、除熱、下氣、祛痰、止渴調中、殺蟲的功效,治肢體痛、肺痨病。烏梅泡水喝能治傷寒煩熱、止吐瀉,與幹姜一起制...

什麽是脂肪粒?如何消除臉部脂肪粒?

什麽是脂肪粒 在我們的臉上總會長一個個像脂肪的小顆粒,弄也弄不掉,而且顔色還是白白的。它既不是粉刺也不是其他的任何痘痘,它就是脂肪粒。 脂肪粒雖然也是由油脂...

網絡安全治理:國家安全保障的主要方向是打擊犯罪,而不是處置和懲罰受害者

來源:中國青年報 新的攻擊方法不斷湧現,黑客幾乎永遠占據網絡攻擊的上風,我們不可能通過技術手段杜絕網絡攻擊。國家安全保障的主要方向是打擊犯罪,而不是處置和懲罰...

河南夫妻在溫嶺網絡直播“造人”內容涉黃被刑事拘留

夫妻網絡直播“造人”爆紅   1月9日,溫嶺城北派出所接到南京警方的協查通告,他們近期打掉了一個涉黃直播APP平台。而根據掌握的線索,其中有一對涉案的夫妻主播...

如何防止牆紙老化?牆紙變舊變黃怎麽辦?

如何防止牆紙老化? (1)選擇透氣性好的牆紙 市場上牆紙的材質分無紡布的、木纖維的、PVC的、玻璃纖維基材的、布面的等,相對而言,PVC材質的牆紙最不透氣...

鮮肌之謎非日本生産VS鮮肌之謎假日貨是謠言

觀點一:破日本銷售量的“鮮肌之謎” 非日本生産 近一段時間,淘寶上架了一款名爲“鮮肌之謎的” 鲑魚卵巢美容液,號稱是最近日本的一款推出的全新護膚品,産品本身所...

中國最美古詩詞精選摘抄

系腰裙(北宋詞人 張先) 惜霜蟾照夜雲天,朦胧影、畫勾闌。人情縱似長情月,算一年年。又能得、幾番圓。 欲寄西江題葉字,流不到、五亭前。東池始有荷新綠,尚小如...

關于女人的經典語句

關于女人的經典語句1、【做一個獨立的女人】 思想獨立:有主見、有自己的人生觀、價值觀。有上進心,永遠不放棄自己的理想,做一份自己喜愛的事業,擁有快樂和成就...

未來我們可以和性愛機器人結婚嗎?

你想體驗機器人性愛嗎?你想和性愛機器人結婚嗎?如果你想,機器人有拒絕你的權利嗎? 近日,第二屆“國際人類-機器人性愛研討會”大會在倫敦金史密斯大學落下帷幕。而...

全球最變態的十個地方

10.土耳其地下洞穴城市 變態指數:★★☆☆☆ 這是土耳其卡帕多西亞的一個著名景點,傳說是當年基督教徒們爲了躲避戰爭而在此修建。裏面曾住著20000人,...

科學家稱,人類死亡後意識將在另外一個宇宙中繼續存活

據英國《每日快報》報道,一位科學家兼理論家Robert Lanza博士宣稱,世界上並不存在人類死亡,死亡的只是身體。他認爲我們的意識借助我們體內的能量生存,而且...

《屏裏狐》片頭曲《我愛狐狸精》歌詞是什麽?

《我愛狐狸精》 - 劉馨棋   (電視劇《屏裏狐》主題曲)   作詞:金十三&李旦   作曲:劉嘉   狐狸精 狐狸仙   千年修...

 
 
 
 C語言中有一種長度不確定的參數,形如:"…",它主要用在參數個數不確定的函數中,我們最輕易想到的例子是PRintf函數。   原型:   int printf( const char *format [, argument]... );   使用例:printf("Enjoy yourself everyday!\n");   printf("The value is %d!\n", value);   這種可變參數可以說是C語言一個比較難理解的部分,這裏會由幾個問題引發一些對它的分析。   注重:在C++中有函數重載(overload)可以用來區別不同函數參數的調用,但它還是不能表示任意數量的函數參數。   問題:printf的實現   請問,如何自己實現printf函數,如何處理其中的可變參數問題? 答案與分析:   在標准C語言中定義了一個頭文件專門用來對付可變參數列表,它包含了一組宏,和一個va_list的typedef聲明。一個典型實現如下:   typedef char* va_list;   #define va_start(list) list = (char*)&va_alist   #define va_end(list)   #define va_arg(list, mode)\   ((mode*) (list += sizeof(mode)))[-1]   自己實現printf:   #include   int printf(char* format, …)   {   va_list ap;   va_start(ap, format);   int n = vprintf(format, ap);   va_end(ap);   return n;   }   問題:運行時才確定的參數   有沒有辦法寫一個函數,這個函數參數的具體形式可以在運行時才確定?   答案與分析:   目前沒有"正規"的解決辦法,不過獨門偏方倒是有一個,因爲有一個函數已經給我們做出了這方面的榜樣,那就是main(),它的原型是:   int main(int argc,char *argv[]);   函數的參數是argc和argv。   深入想一下,"只能在運行時確定參數形式",也就是說你沒辦法從聲明中看到所接受的參數,也即是參數根本就沒有固定的形式。 常用的辦法是你可以通過定義一個void *類型的參數,用它來指向實際的參數區,然後在函數中根據根據需要任意解釋它們的含義。這就是main函數中argv的含義,而argc,則用來表明實際的參數個數,這爲我們使用提供了進一步的方便,當然,這個參數不是必需的。   雖然參數沒有固定形式,但我們必然要在函數中解析參數的意義,因此,理所當然會有一個要求,就是調用者和被調者之間要對參數區內容的格式,大小,有效性等所有方面達成一致,否則南轅北轍各說各話就慘了。   問題:可變長參數的傳遞   有時候,需要編寫一個函數,將它的可變長參數直接傳遞給另外的函數,請問,這個要求能否實現?   答案與分析:   目前,你尚無辦法直接做到這一點,但是我們可以迂回前進,首先,我們定義被調用函數的參數爲va_list類型,同時在調用函數中將可變長參數列表轉換爲va_list,這樣就可以進行變長參數的傳遞了。看如下所示:   void subfunc (char *fmt, va_list argp)   {   ...   arg = va_arg (fmt, argp); /* 從argp中逐一取出所要的參數 */   ...   }   void mainfunc (char *fmt, ...)   {   va_list argp;   va_start (argp, fmt); /* 將可變長參數轉換爲va_list */   subfunc (fmt, argp); /* 將va_list傳遞給子函數 */   va_end (argp);   ...   }   問題:可變長參數中類型爲函數指針   我想使用va_arg來提取出可變長參數中類型爲函數指針的參數,結果卻總是不正確,爲什麽?   答案與分析:   這個與va_arg的實現有關。一個簡單的、演示版的va_arg實現如下:   #define va_arg(argp, type) \   (*(type *)(((argp) += sizeof(type)) - sizeof(type)))   其中,argp的類型是char *。   假如你想用va_arg從可變參數列表中提取出函數指針類型的參數,例如   int (*)(),則va_arg(argp, int (*)())被擴展爲:   (*(int (*)() *)(((argp) += sizeof (int (*)())) -sizeof (int (*)())))   顯然,(int (*)() *)是無意義的。   解決這個問題的辦法是將函數指針用typedef定義成一個獨立的數據類型,例如:   typedef int (*funcptr)();   這時候再調用va_arg(argp, funcptr)將被擴展爲:   (* (funcptr *)(((argp) += sizeof (funcptr)) - sizeof (funcptr)))   這樣就可以通過編譯檢查了。   問題:可變長參數的獲取   有這樣一個具有可變長參數的函數,其中有下列代碼用來獲取類型爲float的實參:   va_arg (argp, float);   這樣做可以嗎?   答案與分析:   不可以。在可變長參數中,應用的是"加寬"原則。也就是float類型被擴展成double;char, short被擴展成int。因此,假如你要去可變長參數列表中原來爲float類型的參數,需要用va_arg(argp, double)。對char和short類型的則用va_arg(argp, int)。   問題:定義可變長參數的一個限制   爲什麽我的編譯器不答應我定義如下的函數,也就是可變長參數,但是沒有任何的固定參數?   int f (...)   {   ...   }   答案與分析:   不可以。這是ANSI C 所要求的,你至少得定義一個固定參數。   這個參數將被傳遞給va_start(),然後用va_arg()和va_end()來確定所有實際調用時可變長參數的類型和值。 [url=/bbs/detail_1785403.html][img]http://image.wangchao.net.cn/it/1323423569863.gif[/img][/url] 更多內容請看C/C++進階技術文檔專題,或
󰈣󰈤
 
 
 
  免責聲明:本文僅代表作者個人觀點,與王朝網路無關。王朝網路登載此文出於傳遞更多信息之目的,並不意味著贊同其觀點或證實其描述,其原創性以及文中陳述文字和內容未經本站證實,對本文以及其中全部或者部分內容、文字的真實性、完整性、及時性本站不作任何保證或承諾,請讀者僅作參考,並請自行核實相關內容。
 
 
陽光靓麗的模特兒(8)
陽光靓麗的模特兒(7)
陽光靓麗的模特兒(6)
陽光靓麗的模特兒(5)
秋-印象
德慶盤龍峽 一
松江印象之三
雲之南(寬幅)
 
>>返回首頁<<
 
 
 
 熱帖排行
 
 
 
 
© 2005- 王朝網路 版權所有