重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
#include stdarg.h //可变参数函数必要的头文件
创新互联建站服务项目包括英山网站建设、英山网站制作、英山网页制作以及英山网络营销策划等。多年来,我们专注于互联网行业,利用自身积累的技术优势、行业经验、深度合作伙伴关系等,向广大中小型企业、政府机构等提供互联网行业的解决方案,英山网站推广取得了明显的社会效益与经济效益。目前,我们服务的客户以成都为中心已经辐射到英山省份的部分城市,未来相信会继续扩大服务区域并继续获得客户的支持与信任!
#include stdio.h
long int sum(unsigned int argc,...) //可变参数函数必须包含至少一个确定的参数
{
long int srt=0;
int va;
unsigned int i;
va_list vl; //用于指向可变参数列表
va_start(vl,argc); //初始化,va_start()调用的第一个参数是所编写的可变参数函数的最后一个确定的参数
for (i = 0; iargc; i++) {
srt+=(int)va_arg(vl,int); //从参数列表按提取一个int类型的值参与相加运算
}
va_end(vl); //清理
return srt;
}
int main(int argc, char* argv[])
{
printf("%ld\n",sum(3,1,2,3));
return 0;
}
这是一个变参函数声明。
加三个点就是了。
取得参数的套路是
这样三步,就将 各个参数,放在了 buf 中。
完整函数如下:
调用如下:
其中,vsprintf 可能造成内存泄漏,因为传入的 buf 的大小未知。
可换成
函数原型:
vsprintf 函数
vsnprintf 函数
#include
void
foo(int
x,
int
y,
int
z)
{
printf("x
=
%d
at
[%x]n",
x,
x);
printf("y
=
%d
at
[%x]n",
y,
y);
printf("z
=
%d
at
[%x]n",
z,
z);
}
int
main(int
argc,
char
*argv[])
{
foo(100,
200,
300);
return
0;
}
运行结果:
x
=
100
at
[bfe28760]
y
=
200
at
[bfe28764]
z
=
300
at
[bfe28768]
c程序栈底为高地址,栈顶为低地址,因此上面的实例可以说明函数参数入栈顺序的确是从右至左的。可到底为什么呢?查了一直些文献得知,参数入栈顺序是和具体编译器实现相关的。比如,pascal语言中参数就是从左到右入栈的,有些语言中还可以通过修饰符进行指定,如visual
c++.即然两种方式都可以,为什么c语言要选择从右至左呢?
进一步发现,pascal语言不支持可变长参数,而c语言支持这种特色,正是这个原因使得c语言函数参数入栈顺序为从右至左。具体原因为:c方式参数入栈顺序(从右至左)的好处就是可以动态变化参数个数。通过栈堆分析可知,自左向右的入栈方式,最前面的参数被压在栈底。除非知道参数个数,否则是无法通过栈指针的相对位移求得最左边的参数。这样就变成了左边参数的个数不确定,正好和动态参数个数的方向相反。
因此,c语言函数参数采用自右向左的入栈顺序,主要原因是为了支持可变长参数形式。换句话说,如果不支持这个特色,c语言完全和pascal一样,采用自左向右的参数入栈方式
有专门的宏,处理可变参
void va_start( va_list arg_ptr, prev_param );
type va_arg( va_list arg_ptr, type );
void va_end( va_list arg_ptr );
一个简单的例子
void simple_va_fun(int i, ...)
{
va_list arg_ptr;
int j=0;
va_start(arg_ptr, i);
j=va_arg(arg_ptr, int);
va_end(arg_ptr);
printf("i=%d j=%d\bn", i, j);
return;
}
int main()
{
simple_va_fun(1);
simple_va_fun(1,2);
simple_va_fun(1,200);
return 0;
}