资源描述:
《c语言中可变参数的用法07638》由会员上传分享,免费在线阅读,更多相关内容在行业资料-天天文库。
1、C语言中可变参数的用法我们在C语言编程中会遇到一些参数个数可变的函数,例如printf()这个函数,它的定义是这样的: intprintf(constchar*format,...); 它除了有一个参数format固定以外,后面跟的参数的个数和类型是可变的,例如我们可以有以下不同的调用方法: printf("%d",i); printf("%s",s); printf("thenumberis%d,stringis:%s",i,s); 究竟如何写可变参数的C函数以及这些可变参数的函数编译器是
2、如何实现的呢?本文就这个问题进行一些探讨,希望能对大家有些帮助.会C++的网友知道这些问题在C++里不存在,因为C++具有多态性.但C++是C的一个超集,以下的技术也可以用于C++的程序中.限于本人的水平,文中如果有不当之处,请大家指正.(一)写一个简单的可变参数的C函数 下面我们来探讨如何写一个简单的可变参数的C函数.写可变参数的C函数要在程序中用到以下这些宏: voidva_start(va_listarg_ptr,prev_param); typeva_arg(va_listarg_ptr,
3、type); voidva_end(va_listarg_ptr); va在这里是variable-argument(可变参数)的意思.这些宏定义在stdarg.h中,所以用到可变参数的程序应该包含这个头文件.下面我们写一个简单的可变参数的函数,改函数至少有一个整数参数,第二个参数也是整数,是可选的.函数只是打印这两个参数的值. voidsimple_va_fun(inti,...) { va_listarg_ptr; intj=0; va_start(arg_ptr,i); j=va_
4、arg(arg_ptr,int); va_end(arg_ptr); printf("%d%d",i,j); return; } 我们可以在我们的头文件中这样声明我们的函数: externvoidsimple_va_fun(inti,...); 我们在程序中可以这样调用: simple_va_fun(100); simple_va_fun(100,200); 从这个函数的实现可以看到,我们使用可变参数应该有以下步骤: 1)首先在函数里定义一个va_list型的变量,这里是arg
5、_ptr,这个变量是指向参数的指针. 2)然后用va_start宏初始化变量arg_ptr,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数. 3)然后用va_arg返回可变的参数,并赋值给整数j.va_arg的第二个参数是你要返回的参数的类型,这里是int型. 4)最后用va_end宏结束可变参数的获取.然后你就可以在函数里使用第二个参数了.如果函数有多个可变参数的,依次调用va_arg获取各个参数. 如果我们用下面三种方法调用的话,都是合法的,但结果却不一样: 1) si
6、mple_va_fun(100); 结果是:100-123456789(会变的值) 2) simple_va_fun(100,200); 结果是:100200 3) simple_va_fun(100,200,300); 结果是:100200 我们看到第一种调用有错误,第二种调用正确,第三种调用尽管结果正确,但和我们函数最初的设计有冲突.下面一节我们探讨出现这些结果的原因和可变参数在编译器中是如何处理的.(二)可变参数在编译器中的处理 我们知道va_start,va_arg,va_en
7、d是在stdarg.h中被定义成宏的,由于1)硬件平台的不同2)编译器的不同,所以定义的宏也有所不同,下面以VC++中stdarg.h里x86平台的宏定义摘录如下(’’号表示折行): typedefchar*va_list; #define_INTSIZEOF(n) ((sizeof(n)+sizeof(int)-1)&~(sizeof(int)-1)) #defineva_start(ap,v)(ap=(va_list)&v+_INTSIZEOF(v)) #defineva_arg(ap
8、,t) (*(t*)((ap+=_INTSIZEOF(t))-_INTSIZEOF(t))) #defineva_end(ap)(ap=(va_list)0) 定义_INTSIZEOF(n)主要是为了某些需要内存的对齐的系统.C语言的函数是从右向左压入堆栈的,图(1)是函数的参数在堆栈中的分布位置.我们看到va_list被定义成char*,有一些平台或操作系统定义为void*.再看va_start的定义,定义为&v+_INTSIZE