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等定義
float sum( float first, ... ) //,...代表可變形參函數
{
float i=first,sum=0;
va_list maker; //va_list 類型資料可以儲存函數的所有參數,做為一個列表一樣儲存
va_start(maker,first); //設定列表的起始位置
while(i!=-1.0)
{
sum+=i;
i=va_arg(maker,float); //返回maker列表的當前值,並指向列表的下一個位置
}
return sum;
}
void main(void)
{
printf( "sum is: %f\n", sum( 2.0,8.0,8.5,-1.0 ) ); //函數調用
}
原文地址: http://blog.163.com/zwh50687695/blog/static/223116332010026112224105/