C语言 可变参数列表 及其局限 以及 printf 函数的深入理解
建立参数个数不定的函数是可能的。如标准库函数printf,它的参数个数是不
定的。虽然printf至少必须用一个字符串作为其第一个参数,但是它能够接收任何个数
的附加参数。printf函数的原型为:
int printf(const char *format, ...);
该函数原型中的省略号(...)表示这个函数接收个数不定的任何类型的参数。注意省略号
必须放在参数列表的最后。
变长参数头文件stdarg.h中的宏定义(见下表)课用来建立变长参数列表。下面给
出的演示程序中给出了接收可变参数个数的函数average。函数average的第一个参数是要
被求平均值的数据的个数。
+-----------------------------------------------------------------------------+
| 在头文件stdarg.h中定义的类型和宏 |
+-----------------------------------------------------------------------------+
| 标识苻 | 解释 |
+----------+------------------------------------------------------------------+
| va_list | 用来保存va_start, va_end所需信息的一种类型。为了访问变长参数列 |
| | 表中的参数,必须声明va_list类型的一个对象 |
+----------+------------------------------------------------------------------+
| va_start | 访问变长参数列表中的参数之前使用的宏,它初始化用va_list声明的 |
| | 的对象,初始化结果供宏va_arg 和va_end使用。 |
+----------+------------------------------------------------------------------+
| va_arg | 展开成一个表达式的宏,该表达式具有变长参数列表中下一个参数的 |
| | 值和类型。每次调用va_arg都会修改用va_list声明的对象,从而使该 |
| | 对象指向参数列表中的下一个参数。 |
+----------+------------------------------------------------------------------+
| va_end | 该宏使程序能够从变长参数列表用宏va_start应用的函数中正常返回 |
+----------+------------------------------------------------------------------+
函数average使用了头文件stdarg.h中所有的年个月和宏。宏va_start, va_arg
和va_end 用va_list类型的对象ap处理函数avergage的变长参数列表。 函数先用
va_start初始化供va_arg 和va_end 使用的对象ap。 宏va_start有两个参数,一个是
对象ap,另各一个是变长参数列表中省略号前的最右边参数的标识符(本例中为i,
va_start用i确定变长参数列表的起始位置)。 然后,函数average反复把变长参数
列表中的参数与变量total相加,加到total中值是用宏va_arg从变长参数列表中检索
到的。宏va_arg有两个参数,一个是对象ap,另一个是要从参数列表中接收的值的类型
(本例中为double),它返回该参数的值。宏va_end只有对象ap这样一个参数。函数
average用宏va_end使得程序从average正常返回到main。最终,函数计算平均值返回到
main。
常见的程
设计
领导形象设计圆作业设计ao工艺污水处理厂设计附属工程施工组织设计清扫机器人结构设计
错误是把省略号放在函数参数列表中间。省略号只能放在参数
列表的最后。那么函数printf和scanf是怎样知道宏va_arg每次使用的类型的呢?
答案
八年级地理上册填图题岩土工程勘察试题省略号的作用及举例应急救援安全知识车间5s试题及答案
是printf和scanf是通过扫描格式控制串中的格式转换苻来确定所要处理的下一个参数
的类型的。
附测试程序:VarParam.c, RedHat Linux 9.0 下测试通过。
#include
#include
double average(int, ...);
main()
{
double w = 37.5, x = 22.5, y = 1.7, z = 10.2;
printf("w = %.1f\nx = %.1f\ny = %.1f\nz = %.1f\n", w, x, y, z);
printf("the average of w and x is : %f.1\n", average(2, w, x));
printf("the average of w, x and y is : %f.1\n", average(3, w, x, y));
printf("the average of w, x, y and z is : %f.1\n", average(4, w, x, y, z));
}
double average(int i, ...)
{
double total = 0;
int j;
va_list ap;
va_start(ap, i);
for (j = 1; j <= i; j++)
total += va_arg(ap, double);
va_end(ap);
return total / i;
}
程序运行输出结果:
w = 37.5
x = 22.5
y = 1.7
z = 10.2
the average of w and x is : 30.000000.1
the average of w, x and y is : 20.566667.1
the average of w, x, y and z is : 17.975000.1
C语言 可变参数列表 及其局限 以及 printf 函数的深入理解在C语言中,要理解printf和scanf函数,就必须理解C语言中可变参数列表的原理。
在C语言中,可变参数列表是利用宏实现的,在 stdarg.h 头文件中定义,是标准库的一部分。
该头文件声明了类型va_list和三个宏va_start、va_arg、va_end。通过它们配合使用,可以访问可变参数列表。
先给例子:计算可变参数列表的参数平均值
#include
float average(int n_values,...)//n_values 是参数个数
{
va_list va;//声明变量
va_start(va,n_values);//初始化
float sum = 0;
int count;
for(count =0;count
内容
财务内部控制制度的内容财务内部控制制度的内容人员招聘与配置的内容项目成本控制的内容消防安全演练内容
来判断(比如,0001可以解读为整数1,也可以解读为某个符号,更可以解读为某个操作,单凭这个数值本身无法知道其类型;C语言[确切说几乎所有编程语言]中,确定一个值类型是由编译器完成的,通过扫描源代码,如果是整型,就生成整型操作执行来操作这个数,如果是符号,就生成符号操作指令来操作这个数。)
1)这些宏无法判断实际存在的参数的数量
2)这些宏无法判断每个参数的类型
所以,对于1),必须给可变参数函数传入可变参数数量 n_values,否则编译器不知道究竟有多少个可变参数,一旦超过实际参数个数,就会读取到垃圾值;一旦少于实际参数个数,后面的参数就会被忽略。
对于2),使用va_arg取值时,必须指定参数类型,否则,编译器不知道该产生什么类型的指令来操作数值(上面说了,值内容都是一串二进制,表达的意思关键在于如何解读,值本身是无法知道类型的)。
--------------------
下面我们来看printf函数:
printf(“%s %d",str,a);
我们知道,printf和scanf支持任意数量参数,其中第一个字符串参数就是用于解决上面两个限制的:通过%s、%d知道了后面有两个可变参数,并且类型一个是字符串、一个是整型。
类似这种原理的还有很多函数,包括C#、Java等语言,如C#中的:
System.Console.WriteLine(" {0},{1}”,str,a);
只不过C#语言的机制允许运行时知道参数的类型(C#元数据中记录了参数的详细信息,这也是C#语言能被完全反编译的问题所在),所以无需%d、%s等的类型限制符。
在C++中,cout和cin其实是封装了的printf。比如cout在输出字符串时,其本质是printf("%s",str),在输出整数时,是print("%d",a);极大地方便了输入输出。(事实上,这也导致了问题,就是在C++中,cin和cout封装的太多,导致学习时难以深入,这也是我建议学习指针等基础时使用C,而不是C++的原因。比如,需要输出str字符串的二进制内容或地址时,就必须借助于强制类型转换;而在C中,只需修改%s为%x。)