C语言可变参数,参数传递
1、函数形参的存储
函数形参在函数中存储是按照栈的方式来存储的
实例:
#include <stdio.h>
void fun(int a0,int a1,int a2, int a3)
{
int *temp;
temp=&a0;
int i;
for(i=0;i<4;i++){
printf("%d\t",*temp++);
}
}
int main(){
fun(1,2,3,4);
getchar();
return 0;
}
函数输出为:
1 2 3 4
2、一般函数的定义在内存中的存储方式
一般的变量定义之后,相同的变量类型存储在相邻的一块内存
例子:
#include <stdio.h>
int main()
{
int a0=1;
char c0='a';
double d0=1.1;
char c1='b';
double d1=2.2;
int a1=2;
int a2=3;
int *pint = &a0;
char *pch = &c0;
double *pd = &d0;
printf("%p:%d\n", pint, *pint);
pint++;
printf("%p:%d\n", pint, *pint);
pint++;
printf("%p:%d\t%p\n", pint, *pint);
printf("%p:%c\n", pch, *pch);
pch++;
printf("%p:%c\n", pch, *pch);
printf("%p:%.2f\n", pd, *pd);
pd++;
printf("%p:%.2f\n", pd, *pd);
}
输出:
0xbf944c54:1
0xbf944c58:2
0xbf944c5c:3
0xbf944c6e:a
0xbf944c6f:b
0xbf944c40:1.10
0xbf944c48:2.20
**这个结论只是对于我自己的编译器的结果,具体可能对于另外一个编译器的输出有有可能不一样,而且,如果把有变量没有使用过的话,编译器有可能丢弃它(自己的猜想而已,等过阵子研究《深入理解计算机系统》的时候在研究这个问题)。
3、C语言给我们的可变形参va_list
头文件stdarg.h
下面定义了四个宏:
1. va_list
在stdarg.h中定义:typedef char * va_list;
2. void va_start(va_list ap,last);
这是第一个调用的宏,last是可变参数的前一个确定的参数。因此只有可变参数的函数是不允许的,如fun(...)是不合法的。
#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) ) //第一个可选参数地址
3. void va_arg(va_list ap, type);
返回当前参数并且使ap指向下一个参数,type是明确的类型名,就是获取的类型,不是。
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ) //下一个参数地址
4. void va_end(va_list ap);
有va_start就必须有va_end,作用是使释放ap
#define va_end(ap) ( ap = (va_list)0 ) // 将指针置为无效
4. void va_copy(va_list dest, va_list src);
每次调用va_copy时也必须要有va_end调用。
例子:
#include <stdio.h>
#include <stdarg.h>
void fun(int a,...){
va_list ap;
va_start(ap, a);
int i;
for(i=0; i<a; i++){
printf("%d\t",va_arg(ap,int));
}
putchar('\n');
va_end(ap);
}
int main(){
fun(4,1,2,3,4);
printf("--------------------------------\n");
fun(4,1,2,"ab","cd");
return 0;
}
输出:
1 2 3 4
--------------------------------
1 2 134514184 134514181