通过《C程序设计语言》第二版第7章的介绍,我们知道标题准确翻译或许是“变长变元表”,这个无可厚非,我们只要知道我们要编写一个函数参数个数不确定的函数,如何编写?
形式:
int printf(char *fmt, ...);
从形式上出发,我们需要处理的部分是”…”。
这部分的处理,我们可以这样设计:
首先“…”表示的是一个变元表,这个表我们可以理解成是一个链表,对于链表的处理,我们往往需要一个指针指向当前元素。变元表的遍历可以采用链表的遍历思路。
1.获得变元表的开始位置
2.获得当前位置的元素值,并且位置+1处理
3.循环执行步骤2,直到变元表结束
这里有个问题,我们并不知道,变元表的结束位置。也就是说,我们必须知道我们的变元表的元素个数,这样上面的设计思路才能实施,假设我们知道变元表中的元素个数,这也是编写变长变元表的前提。
我们现在使用文件 stdarg.h 文件中函数,实现变元表的处理:
va_list: 用于定义变元变量指针
va_start:初始化变元变量指针
va_end:释放处理过程中的资源
va_arg:获得当前变元变量指针对应的值(依赖数据类型),并且+1变元变量的指针值
从这个设计我们可以看出,对于表的处理,基本上就是“定界遍历”这样的方式。这些都是基本的遍历思想。我们从这里也明白,对于计算机的理解要建立在最基础的理解方式上面,因为,这样的理解是深刻的,没有特殊的元素在这里面,都是些基本的设计理念。
一个简单的例子:
/*************************************************************************
> File Name: main.c
> Author: mhsheng
> Mail:981065720@qq.com
> Created Time: Mon 14 Nov 2016 03:14:04 PM CST
************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
void
minprintf(char *fmt, ...)
{
va_list ap;
char *p;
int iVal;
double dVal;
va_start(ap, fmt);
for (p=fmt; *p!='\0'; ++p) {
if (*p != '%') {
putchar(*p);
continue;
}
++p;
switch (*p) {
case 'd':
iVal = va_arg(ap, int);
printf("%d", iVal);
break;
case 'f':
dVal = va_arg(ap, double);
printf("%f", dVal);
break;
default:
break;
}
}
va_end(ap);
}
int
main(int argc, char **argv)
{
minprintf("%d, %f\n", 13, 0.12);
exit(0);
}