Posted on 2009-01-13 11:10
Prayer 閱讀(620)
評論(0) 編輯 收藏 引用 所屬分類:
C/C++
◎用法:
func( Type para1, Type para2, Type para3, ... )
{
/****** Step 1 ******/
va_list ap;
va_start( ap, para3 ); //一定要“...”之前的那個參數
/****** Step 2 ******/
//此時ap指向第一個可變參數
//調用va_arg取得里面的值
Type xx = va_arg( ap, Type );
//Type一定要相同,如:
//char *p = va_arg( ap, char *);
//int i = va_arg( ap, int );
//如果有多個參數繼續調用va_arg
/****** Step 3 ******/
va_end(ap); //For robust!
}
◎研究:
typedef char * va_list;
#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_list argptr;
C語言的函數是從右向左壓入堆棧的,調用va_start后,
按定義的宏運算,_ADDRESSOF得到v所在的地址,然后這個
地址加上v的大小,則使ap指向第一個可變參數如圖:
棧底 高地址
| .......
| 函數返回地址
| .......
| 函數最后一個參數
| ....
| 函數第一個可變參數 <--va_start后ap指向
| 函數最后一個固定參數
| 函數第一個固定參數
棧頂 低地址
然后,用va_arg()取得類型t的可變參數值, 先是讓ap指向下一個參數:
ap += _INTSIZEOF(t),然后在減去_INTSIZEOF(t),使得表達式結果為
ap之前的值,即當前需要得到的參數的地址,強制轉換成指向此參數的
類型的指針,然后用*取值
最后,用va_end(ap),給ap初始化,保持健壯性。
example:(chenguiming)
#include <stdio.h>
#include <ctype.h>
#include<stdlib.h>
#include <stdarg.h>
int average( int first, ... ) //變參數函數,C++里也有
{
int count=0,i=first,sum=0;
va_list maker; //va_list 類型數據可以保存函數的所有參數,做為一個列表一樣保存
va_start(maker,first); //設置列表的起始位置
while(i!=-1)
{
sum+=i;
count++;
i=va_arg(maker,int);//返回maker列表的當前值,并指向列表的下一個位置
}
return sum/count;
}
void main(void)
{
printf( "Average is: %d\n", average( 2, 3, 4,4, -1 ) );
}