由于在C语言中没有函数重载,解决不定数目函数参数问题变得比较麻烦;即使采用C++,如果参数个数不能确定。也很难采用函数重载。对这种情况,有些人采用指针参数来解决问题。本文就C语言中对不定参数函数的使用方法进行小结。
1、函数参数在堆栈中的分布
地址从高到低,依次是:函数参数列表,函数返回地址,函数执行代码段。堆栈中,各个函数的分布情况是倒序的,即最后一个参数在列表中地址最高部分,第一个参数在列表地址的最低部分。
2、VA_LIST宏是在C语言中解决变参问题的一组宏,在C语言的stdarg.h头文件中有一下几个定义:
(1)_INTSIZEOF 宏,获取类型占用的空间长度,最小占用长度为int的整数倍
#define _INTSIZEOF(n) ((sizeof(n)+sizeof(int)-1)&~(sizeof(int) - 1) )
(2)VA_START宏,获取可变参数列表的第一个参数的地址(ap是类型为va_list的指针,v是可变参数最左边的参数
#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )
(3)VA_ARG宏,获取可变参数的当前参数,返回指定类型并将指针指向下一参数(t参数描述了当前参数的类型)
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
(4)VA_END宏,清空va_list可变参数列表
#define va_end(ap) ( ap = (va_list)0 )
3、VA_LIST使用方法:
(1)首先在函数里定义一具VA_LIST型的变量,这个变量是指向参数的指针;
(2)然后用VA_START宏初始化变量刚定义的VA_LIST变量;
(3)然后用VA_ARG返回可变的参数,VA_ARG的第二个参数是你要返回的参数的类型(如果函数有多个可变参数的,依次调用VA_ARG获取各个参数);
(4)最后用VA_END宏结束可变参数的获取。
4、使用可变参数需要注意的问题:
(1)可变参数的类型和个数完全由程序代码控制,它并不能智能地识别不同参数的个数和类型;
(2)如果我们不需要一一详解每个参数,只需要将可变列表拷贝至某个缓冲,可用vsprintf函数;
(3)因为编译器对可变参数的函数的原型检查不够严格,对编程查错不利。不利于我们写出高质量的代码。
(4) 可变参数是由宏实现的,但是由于硬件平台的不同,编译器的不同,宏的定义也不相同。
5、例子:
(1)打印字符串
(2)求和
(3)使用vsprintf将可变参数全部写入缓冲区
函数声明:int vsprintf(char *buf,const char *format, va_list arglist);
函数用途:该函数作用同sprintf函数,区别是参数表由一个va_list类型的指针代替
头文件 :stdio.h stdarg.h
下例摘自S3C2440中的串口打印函数