[转][C/C++] 函数中不定长参数的使用

转自http://book.51cto.com/art/200902/109024.htm

http://hi.baidu.com/zzzkkk666/blog/item/683ac15472d29e5cd10906dd.html

5.4.1. 变长参数的使用

5.4 函数的变长参数

文件的格式化输入输出函数都支持变长参数。定义时,变长参数列表通过省略号"…"表示,因此,具有变长参数列表的函数定义格式为:

type 函数名(参数1, 参数2, 参数n, ...);

其中type为函数的返回值类型,参数1~参数n为定长参数,"..."代表变长参数,注意"..."必须定义在参数的最右端。如下例:

int printf(const char *format, ...);
int mysum(...);

5.4.1. 变长参数的使用

Unix的变长参数通过va_list对象实现,定义在文件"stdarg.h"中,变长参数的应用模板如代码5-15所示:

代码5-15 变长参数代码模板

#include <stdarg.h>
function (parmN, ...)
va_list pvar;
……………………………
va_start (pvar, parmN);
while()
{
……………………
f = va_arg (pvar, type);
……………………
}
va_end (pvar);

1. va_list pvar

申明va_list数据类型变量pvar,该变量访问变长参数列表中的参数。

2. va_start(pvar, parmN)

宏va_start初始化变长参数列表。pvar是va_list型变量,在步骤1中定义,记载列表中的参数信息。parmN是省略号"..."前的一个参数名,va_start根据此参数,判断参数列表的起始位置,如:

例1. 函数:function(parmN, …)

答:va_start(pvar, parmN);

例2. 函数:int mysum(int i, int j, …)

答:va_start(pvar, j);

3. va_arg(pvar, type)

获取变长参数列表中参数的值。pvar是步骤1中定义的va_list型变量,type为参数值的类型,也是宏va_arg返回数值的类型,如:

va_arg(pvar, int);   /* 将参数列表中的当前参数值转化为int型返回 */
va_arg(pvar, float);  /* 将参数列表中的当前参数值转化为float型返回*/

宏va_arg执行完毕后自动更改对象pvar,将其指向下一个参数。

4. va_end(pvar)

关闭本次对变长参数列表的访问。

实例

设计函数mysum,计算输入参数的和并返回结果。源程序如代码5-16所示:

代码5-16 变长参数函数实例(节自/code/chapter5/mysum.c)

#include <stdarg.h>
int mysum(int i, ...)  /* 参数i表明变长参数的个数 */
{
int r=0, j=0;
va_list pvar;
va_start(pvar, i);
for (j=0; j<i; j++)
{
r += va_arg(pvar, int);
}
va_end(pvar);
return(r);
}
void main()
{
printf("sum(1,4)=%d\n", mysum(1, 4));
printf("sum(2,4,8)=%d\n", mysum(2, 4, 8));
}

编译与运行代码5-16:

# make mysum
cc -O -o mysum mysum.c
# ./mysum
sum(1,4)=4
sum=(2,4,8)=12

 

 

5.4.2 变长参数的传递

5.4.2 变长参数的传递

上一节讲述了如何创建具有变长参数的函数和如何读取变长参数,其操作都在函数内完成,本节将讲述把变长参数列表整体作为参数传递给其他函数的方法。

变长参数传递的函数族如下:

#include <stdarg.h>
int vprintf(const char *format, va_list ap);
int vfprintf(FILE *stream, const char *format, va_list ap);
int vsprintf(char *str, const char *format, va_list ap);

这些函数完全等价于格式化输出函数,只是在形式上采用固定参数代替变长参数,这样描述的函数更加紧凑,这些函数常应用于变长参数函数内部的功能实现。

实例

设计函数"int PrintLog(FILE* stream, const char* pformat, ...)",它按照字符串format的内容,控制后继参数的数量和格式,并在文件流stream中输出。源程序如代码5-17所示:

代码5-17 传递变长参数实例(节自/code/chapter5/print1.c)

#include <stdarg.h>
#include <stdio.h>
int PrintLog(FILE* pfile, const char * pformat, ...)
{
va_list _va_list;
char szBuf[1024];
if (pformat == NULL || pfile == NULL) return -1;  /* 判断指针是否正确*/
va_start(_va_list, pformat);      /* 初始化变长参数列表 */
vsprintf(szBuf, pformat, _va_list);    /* 传递变长参数 */
va_end(_va_list);         /* 结束使用变长参数列表 */
fputs(szBuf, pfile);        /* 输出到文件流 */
return 0;
}
void main()
{
PrintLog(stderr, "[%s][%s][%d][%c]\n", "This", "Is", 5, 'a');
PrintLog(stderr, "Error[%p][%.2f][%X]\n", NULL, 3.123, 100);
}

编译与运行代码5-17:

# make print1
cc -O print1.c  -o print1
# ./print1    
[This][Is][5][a]
Error[00000000][3.12][64]

【实践经验】对于指针类型的参数,最好在函数入口处判断其是否为空,以免空指针引用错误。如代码5-17中黑体部分。

 

C语言中可变参数的宏
2007年12月24日 星期一 21:09
    今天来说说宏。什么?宏也能可变参数?是的,你没有听错,带参数的宏和函数一样,同样支持可变参数。下面通过一个小程序加以说明。 
#include 
#include

#define OUTSCREEN(msg, ...) printf(msg,__VA_ARGS__)

int main(int argc, char* argv[])
{
     OUTSCREEN("Hello World!n%s", "__This is a MACRO!n");
     return 0;
}

     这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。

#define OUTSCREEN(msg, ...) printf(msg, __VA_ARGS__)

int main(int argc, char* argv[])
{
     OUTSCREEN("Hello World!n%s", "__This is a MACRO!n");
     return 0;
}

    假如我们将上面的代码稍作一下修改,变成下面的样子。

#define OUTSCREEN(msg, ...) printf(msg, __VA_ARGS__)

int main(int argc, char* argv[])
{
     OUTSCREEN("Hello World!");
     return 0;
}

    注意我仅仅是将main函数里的OUTSCREEN做了修改,这时可变参数的个数为0了。但是编译的时候gcc却报错:
In function `main': 
error: parse error before ')' token


    什么原因导致出错呢?把宏展开一下看看,原来是","惹得祸。那么这种参数个数可以为0的宏要怎么写呢?C99的规范没有定义这个,gcc对此做了扩展。重新定义OUTSCREEN宏如下:
#define OUTSCREEN(msg, ...) printf(msg, ##__VA_ARGS__)
    当可变参数的个数为0时,这里的##起到把前面多余的","去掉,实际上变成了printf(msg),这样编译就能通过了。

    另外,__VA_ARGS__这个宏实在不利于记忆,gcc对此做了扩展,另一种可接受的定义方法为:
#define OUTSCREEN(msg, args...) printf(msg, ##args)

贴上代码,权当练习之用

#include <stdio.h>

//#define OUTSCREEN(msg, ...) printf(msg, __VA_ARGS__)
//#define OUTSCREEN(msg, ...) printf(msg, ##__VA_ARGS__)
#define OUTSCREEN(msg, args...) printf(msg, ##args)
int main(int argc, char* argv[])
{
OUTSCREEN("Hello World=====%s\n");
//OUTSCREEN("Hello World=====%s", "__This is an example\n");
//    OUTSCREEN("Hello World=====%s", "__This is an example\n", "77777777777");
     return 0;
}
 

 

 

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值