C/C++中的可变形参

来源:互联网 发布:查看端口号命令 编辑:程序博客网 时间:2024/06/04 19:01

C语言中的printf,scanf就是最常见的可变形参函数,定义一个可变形参的函数很简单,如void print(int n, ...)   ,函数中对参数的处理主要是通过对栈进行操作,而c函数的实参都是自右向左压入栈的. 主要的栈操作(都是)有va_list,va_start ,va_arg,va_end, 定义如下:

      typedef char * va_list;
      #define _INTSIZEOF(n)   ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
      #define va_start _crt_va_start
      #define va_arg _crt_va_arg
      #define va_end _crt_va_end
      #define _crt_va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
      #define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
      #define _crt_va_end(ap) ( ap = (va_list)0 )

     va_start(ap,v):主要是获取可变参数列表的首地址,然后赋值给ap,近似ap=&v+sizeof(v) (这里暂不考虑内存对齐和类型转换)

      va_arg(ap,t):取得返回类型t的可变参数值, 并使ap指向下一个参数: ap += sizeof(t),这里的t是可变参数的数据类型,如int,float之类

     va_end(ap):给ap初始化

     va_start(ap,v) va_arg(ap,t) va_end(ap)三者合用,保证程序的健壮性.


    一个使用可变形参的简单程序:

#include <stdio.h>
#include <stdarg.h>        //包含va_list等定义

double

sum( double first, ... )   //,...代表可变形参函数
{
  double i=first,sum=0;
  va_list maker;            //va_list 类型数据可以保存函数的所有参数,做为一个列表一样保存
  va_start(maker,first); //设置列表的起始位置
  while(i!=-1.0)
  {
     sum+=i;
     i=va_arg(maker,double); //返回maker列表的当前值,并指向列表的下一个位置
  }
  return sum;

}

void main(void)
{
  printf( "sum is: %f\n", sum( 2.0,8.0,8.5,-1.0 ) );   //函数调用
}