欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C/C++中可變參數(shù)的用法詳細(xì)解析

 更新時間:2013年09月22日 09:08:14   投稿:jingxian  
可變參數(shù)的使用方法遠(yuǎn)遠(yuǎn)不止以下介紹的幾種,不過在C,C++中使用可變參數(shù)時要小心,在使用printf()等函數(shù)時傳入的參數(shù)個數(shù)一定不能比前面的格式化字符串中的’%’符號個數(shù)少,否則會產(chǎn)生訪問越界,運氣不好的話還會導(dǎo)致程序崩潰

可變參數(shù)即表示參數(shù)個數(shù)可以變化,可多可少,也表示參數(shù)的類型也可以變化,可以是int,double還可以是char*,類,結(jié)構(gòu)體等等??勺儏?shù)是實現(xiàn)printf(),sprintf()等函數(shù)的關(guān)鍵之處,也可以用可變參數(shù)來對任意數(shù)量的數(shù)據(jù)進(jìn)行求和,求平均值帶來方便(不然就用數(shù)組或每種寫個重載)。在C#中有專門的關(guān)鍵字parame,但在C,C++并沒有類似的語法,不過幸好提供這方面的處理函數(shù),本文將重點介紹如何使用這些函數(shù)。
 
第一步 可變參數(shù)表示
用三個點…來表示,查看printf()函數(shù)和scanf()函數(shù)的聲明:
int printf(const char *, ...);
int scanf(const char *, ...);

這三個點用在宏中就是變參宏(Variadic Macros),默認(rèn)名稱為__VA_ARGS__。如:
#define WriteLine(...) { printf(__VA_ARGS__); putchar('\n');}
再WriteLine("MoreWindows");

考慮下printf()的返回值是表示輸出的字節(jié)數(shù)。將上面宏改成:
#define WriteLine (...) printf(__VA_ARGS__) + (putchar('\n') != EOF ? 1: 0);
這樣就可以得到WriteLine宏的返回值了,它將返回輸出的字節(jié)數(shù),包括最后的’\n’。如下例所示i和j都將輸出12。

復(fù)制代碼 代碼如下:

       int i = WriteLine("MoreWindows");
       WriteLine("%d", i);
       int j = printf("%s\n", "MoreWindows");
       WriteLine("%d", j);

第二步 如何處理va_list類型
函數(shù)內(nèi)部對可變參數(shù)都用va_list及與它相關(guān)的三個宏來處理,這是實現(xiàn)變參參數(shù)的關(guān)鍵之處。
在<stdarg.h>中可以找到va_list的定義:
typedef char *  va_list;
再介紹與它關(guān)系密切的三個宏要介紹下:va_start(),va_end()和va_arg()。

同樣在<stdarg.h>中可以找到這三個宏的定義:
#define va_start(ap,v)  ( ap = (va_list)&v + _INTSIZEOF(v) )
#define va_end(ap)      ( ap = (va_list)0 )
#define va_arg(ap,t)    ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
其中用到的_INTSIZEOF宏定義如下:
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )

來分析這四個宏:
va_end(ap)這個最簡單,就是將指針置成NULL。
va_start(ap,v)中ap = (va_list)&v + _INTSIZEOF(v)先是取v的地址,再加上_INTSIZEOF(v)。_INTSIZEOF(v)就有點小復(fù)雜了。( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )全是位操作,看起來有點麻煩,其實不然,非常簡單的,就是取整到sizeof(int)。比如sizeof(int)為4,1,2,3,4就取4,5,6,7,8就取8。對x向n取整用C語言的算術(shù)表達(dá)就是((x+n-1)/n)*n,當(dāng)n為2的冪時可以將最后二步運算換成位操作——將最低 n - 1個二進(jìn)制位清 0就可以了。

va_arg(ap,t)就是從ap中取出類型為t的數(shù)據(jù),并將指針相應(yīng)后移。如va_arg(ap, int)就表示取出一個int數(shù)據(jù)并將指針向移四個字節(jié)。

因此在函數(shù)中先用va_start()得到變參的起始地址,再用va_arg()一個一個取值,最后再用va_end()收尾就可以解析可變參數(shù)了。
 
第三步 vfprintf()函數(shù)和vsprintf()函數(shù)
vfprintf()這個函數(shù)很重要,光從名字上看就知道它與經(jīng)常使用的printf()函數(shù)有很大的關(guān)聯(lián)。它有多個重載版本,這里講解最常用的一種:
函數(shù)原型

復(fù)制代碼 代碼如下:

int vfprintf(

   FILE *stream,

   const char *format,

   va_list argptr

);


第一個參數(shù)為一個FILE指針。FILE結(jié)構(gòu)在C語言的讀寫文件必不可少。要對屏幕輸出傳入stdout。
第二個參數(shù)指定輸出的格式。
第三個參數(shù)是va_list類型,這個少見,但其實就是一個char*表示可變參參數(shù)的起始地址。
返回值:成功返回輸出的字節(jié)數(shù)(不包括最后的’\0’),失敗返回-1。
vsprintf()與上面函數(shù)類似,就只列出函數(shù)原型了:
復(fù)制代碼 代碼如下:

int vsprintf(

   char *buffer,

   const char *format,

   va_list argptr

);


還有一個int _vscprintf(const char *format, va_list argptr );可以用來計算vsprintf()函數(shù)中的buffer字符串要多少字節(jié)的空間。

代碼范例
下面就給出了自己實現(xiàn)的printf()函數(shù)(注1)與WriteLine()函數(shù)
 

復(fù)制代碼 代碼如下:

 int Printf(char *pszFormat, ...) 
{
       va_list   pArgList;
      
       va_start(pArgList, pszFormat);
       int nByteWrite = vfprintf(stdout, pszFormat, pArgList);
       va_end(pArgList);
 
       return nByteWrite;
}
 
int WriteLine(char *pszFormat, ...)
{
       va_list   pArgList;
      
       va_start(pArgList, pszFormat);
       int nByteWrite = vfprintf(stdout, pszFormat, pArgList);
       if (nByteWrite != -1)
              putchar('\n'); //注2
       va_end(pArgList);
      
       return (nByteWrite == -1 ? -1 : nByteWrite + 1);
}

調(diào)用與printf()函數(shù)相同。
再給出一個用可變參數(shù)來求和,遺憾的在C,C++中無法確定傳入的可變參數(shù)的個數(shù)(printf()中是通過掃描'%'個數(shù)來確實參數(shù)的個數(shù)的),因此要么就要指定個數(shù),要么在參數(shù)的最后要設(shè)置哨兵數(shù)值:
設(shè)置哨兵數(shù)值:
復(fù)制代碼 代碼如下:

const int GUARDNUMBER = 0; //哨兵標(biāo)識
//變參參數(shù)的個數(shù)無法確定,在printf()中是通過掃描'%'個數(shù),在這通過設(shè)置哨兵標(biāo)識來確定變參參數(shù)的終止
int MySum(int i, ...)
{
       int sum = i;
       va_list argptr;
      
       va_start(argptr, i);
       while ((i = va_arg(argptr, int)) != GUARDNUMBER)
              sum += i;
       va_end(argptr);
      
       return sum;
}

可以這樣的調(diào)用:   printf("%d\n", MySum(1, 3, 5, 7, 9, 0));
但不可以直接傳入一個0:   printf("%d\n", MySum(0)); //error
指定個數(shù):
復(fù)制代碼 代碼如下:

int MySum(int nCount, ...)
{
       if (nCount <= 0)
              return 0;
 
       int sum = 0;
       va_list argptr;
      
       va_start(argptr, nCount);
       for (int i = 0; i < nCount; i++)
              sum += va_arg(argptr, int);
       va_end(argptr);
      
       return sum;
}

調(diào)用時第一個參數(shù)表示后面參數(shù)的個數(shù)如:
       printf("%d\n", MySum(5, 1, 3, 5, 7, 9));
       printf("%d\n", MySum(0));

代碼所用的頭文件:
#include <stdarg.h>
#include <stdio.h>
 
可變參數(shù)的使用方法遠(yuǎn)遠(yuǎn)不止上述幾種,不過在C,C++中使用可變參數(shù)時要小心,在使用printf()等函數(shù)時傳入的參數(shù)個數(shù)一定不能比前面的格式化字符串中的’%’符號個數(shù)少,否則會產(chǎn)生訪問越界,運氣不好的話還會導(dǎo)致程序崩潰。
 
可變參數(shù)的原形理涉及到調(diào)用函數(shù)時參數(shù)的入棧問題,這個下次再開一篇進(jìn)行專門的探討。
 
注1.網(wǎng)上有不用vfprintf()自己解析參數(shù)來實現(xiàn)printf()的,但很少能將功能做到與printf()相近(實際上能完全熟悉printf()的人已經(jīng)就不多,不信的話可以先看看《C陷阱與缺陷》了解printf()很多不太常用的參數(shù),再去Microsoft Visual Studio\VC98\CRT\SRC中查看OUTPUT.C對printf()的實現(xiàn))。

注2.如果輸出單個字符 putchar(ch)會比printf(“%c”, ch)效率高的多。在字符串不長的情況下,多次調(diào)用putchar()也會比調(diào)用printf(“%s\n”, szStr);的效率高。在函數(shù)大量調(diào)用時非常明顯。

相關(guān)文章

  • windows 下C++生成Dump調(diào)試文件與分析

    windows 下C++生成Dump調(diào)試文件與分析

    dump文件是C++程序發(fā)生異常時,保存當(dāng)時程序運行狀態(tài)的文件,是調(diào)試異常程序重要的方法,所以程序崩潰時,除了日志文件,dump文件便成了我們查找錯誤的最后一根救命的稻草,這篇文章主要介紹了windows 下C++生成Dump調(diào)試文件與分析,需要的朋友可以參考下
    2023-04-04
  • Qt中TableView與TreeView組件聯(lián)動實現(xiàn)

    Qt中TableView與TreeView組件聯(lián)動實現(xiàn)

    本文主要介紹了Qt中TableView與TreeView組件聯(lián)動實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-12-12
  • C++ deque/queue/stack的底層原理解析

    C++ deque/queue/stack的底層原理解析

    這篇文章主要介紹了C++ deque/queue/stack的底層原理解析,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-07-07
  • c++制作的時間函數(shù)類

    c++制作的時間函數(shù)類

    本文給大家分享的是一個個人使用C++編寫的時間函數(shù)類,主要是實現(xiàn)了類的定義和調(diào)用,相比較來說還算比較復(fù)雜的時間類了,推薦給小伙伴們,有需要的朋友可以參考下。
    2015-03-03
  • 隨機加密程序的實現(xiàn)方法

    隨機加密程序的實現(xiàn)方法

    下面實例是對隨機加密程序的實現(xiàn)方法。需要的朋友參考下
    2013-05-05
  • C++回溯算法深度優(yōu)先搜索舉例分析

    C++回溯算法深度優(yōu)先搜索舉例分析

    回溯在迷宮搜索中使用很常見,就是這條路走不通,然后返回前一個路口,繼續(xù)下一條路。回溯算法說白了就是窮舉法,下面讓我們一起來看看回溯算法深度優(yōu)先搜索吧
    2022-03-03
  • C++中delete和delete[]的區(qū)別

    C++中delete和delete[]的區(qū)別

    這篇文章主要介紹了C++中delete和delete[]的區(qū)別的相關(guān)資料,需要的朋友可以參考下
    2016-03-03
  • C語言實現(xiàn)打印數(shù)組以及打印注意事項說明

    C語言實現(xiàn)打印數(shù)組以及打印注意事項說明

    這篇文章主要介紹了C語言實現(xiàn)打印數(shù)組以及打印注意事項說明,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • 深入分析C中不安全的sprintf與strcpy

    深入分析C中不安全的sprintf與strcpy

    本篇文章是對C中不安全的sprintf與strcpy函數(shù)的使用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • QT編寫簡單登錄界面的實現(xiàn)示例

    QT編寫簡單登錄界面的實現(xiàn)示例

    登陸界面是網(wǎng)頁中常見的界面,本文主要介紹了QT編寫簡單登錄界面的實現(xiàn)示例,具有一定的參考價值,感興趣的可以了解一下
    2024-02-02

最新評論