转自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;
}
分享到:
相关推荐
例如,一个使用可变参数模板的C++函数可以写成这样: ```cpp #include #include <cstdarg> // C风格可变参数处理 template int sum(Args... args) { return (args + ...); // C++17 折叠表达式简化了可变参数的...
在C/C++编程语言中,函数参数的处理方式有两方面值得关注:参数的计算顺序和参数的压栈顺序。这两个概念对于理解函数调用时的行为至关重要。 首先,我们来看参数的压栈顺序。C/C++标准规定,函数参数在调用时按照从...
不定参数函数是C/C++语言中一类特殊的函数,它们允许函数接收不确定数量的参数。在C/C++标准库中,这种类型的函数并不多见,但以printf()和scanf()系列函数为代表的输入输出函数以及exec*()系列函数执行外部程序时...
- **参数规则**:完整写出参数及其类型,合理命名,避免过多参数,使用`const`修饰只读指针,避免不定参数。 - **返回值规则**:明确返回值类型,返回值与函数名相符,不混杂正常值和错误标志,有时为灵活性考虑可...
C/C++的可变参数表允许函数接受数量不定的参数,这在实现日志打印、错误报告等函数时非常有用。然而,使用可变参数表需要谨慎,因为它可能导致类型安全问题和性能开销。 对于数组名和指针的区别,它们在C/C++中虽然...
传统C++头文件是C++语言的早期版本中使用的头文件,虽然已经被标准C++头文件取代,但仍然被一些旧的编译器和项目所使用。这些头文件包括: * `<fstream.h>`:文件输入/输出流 * `<iomanip.h>`:参数化输入/输出流 *...
C语言中的va函数,也就是可变参数函数(variable argument function),允许函数接收不定数量的参数,这种机制为函数调用提供了极大的灵活性。在C/C++标准库中,printf()和scanf()系列函数就是典型的可变参数函数,...
在编程中,不定参数函数是一种特殊的函数,其参数个数是不确定的。这类函数在C或C++语言中广泛应用于需要传递不确定数量参数的场景。为了编写这样的函数,需要使用一种特殊的机制来处理可变数量的参数。在C语言中,...
不定参数函数实现var_arg系列的宏 你一定要搞明白的C函数调用方式与栈原理 深入理解C/C++中的指针 详解C++11中的智能指针 C++17结构化绑定 C++必须掌握的pimpl惯用法 用Visual Studio调试Linux程序 如何使用Visual ...
不定参数函数是编程语言中的一种特性,允许我们定义可以接受任意数量参数的函数。这种功能在处理各种场景时非常有用,比如需要一个函数能够灵活地处理任意数量的输入数据,或者在不知道具体参数数量的情况下创建通用...
`void`关键字在C/C++中扮演着特殊角色,主要用于声明无类型或未知类型的函数、指针等。`void`指针可以指向任何类型的对象,但必须显式转换为具体类型后才能访问。在嵌入式编程中,`void`指针常用于通用回调函数或...
不定参数在C语言中的应用实例:不定参数当年做为C/C++语言一个特长被很多人推崇,但是实际上这种技术并没有应用很多。除了格式化输出之外,我实在没看到多少应用。主要原因是这种技术比较麻烦,副作用也比较多,而...
C语言支持可变参数列表,通过`stdarg.h`头文件中的宏和函数,可以处理不定数量的参数。掌握变参函数的编写和调用规则,对于编写灵活的函数接口非常有用。 #### 浅谈Unix和Linux下的int 在Unix和Linux环境下,`int`...
C/C++ 支持可变参数函数,即函数可以接受数量不定的参数。这种特性在某些情况下非常有用,例如实现日志记录功能。 - **标准库函数**:`printf` 和 `va_list` 等函数和类型为处理可变参数提供了支持。 - **实现方法*...
变长参数函数为C语言提供了类似于C++函数重载的功能,但是它们之间存在本质的区别。在C++中,函数重载是通过编译器识别不同函数签名来实现的,而C语言的变长参数函数则需要程序员使用预处理宏来手动处理参数。这种...
不定参数在C语言中的应用实例:不定参数当年做为C/C++语言一个特长被很多人推崇,但是实际上这种技术并没有应用很多。除了格式化输出之外,我实在没看到多少应用。主要原因是这种技术比较麻烦,副作用也比较多,而...