轉至 http://www.cnblogs.com/hanyonglu/archive/2011/05/07/2039916.html
本文主要介紹va_start和va_end的使用及原理。
介紹這兩個宏之前先看一下C中傳遞函數的參數時的用法和原理:
1.在C中,當我們無法列出傳遞函數的所有實參的類型和數目時,可以用省略號指定參數表
void foo(...);
2.函數參數的傳遞原理
函數參數是以數據結構:棧的形式存取,從右至左入棧。
首先是參數的內存存放格式:參數存放在內存的堆棧段中,在執行函數的時候,從最後一個開始入棧。因此棧底高地址,棧頂低地址,舉個例子如下: for (int i = 0; i < a; ++i)
{
cout << *temp << endl;
temp++;
}
}
int main()
{
int a = 1;
int b = 2;
int c = 3;
int d = 4;
fun(4, a, b, c, d);
system("pause");
return 0;
}
Output::
1
2
3
4
4.演示如何使用參數個數可變的函數,采用ANSI標准形式
以上是對va_start和va_end的介紹。
大哥,你審題不仔細,被騙了~~~~~
for(;i;i&=i-1);//注意:這個語句後面有一個;,也就是說是個閉循環
所以++k也就是在for(;j<n;++j)每完成一個循環後加1,
for(;j<n;++j){ //這裡循環兩輪對吧,j=1和2。
你自己都算出來,這個循環循環兩次,所以k最後為2
#include <stdarg.h> // 必須包含的頭文件
int Add(int start,...) // ...是作為占位符
{
va_list arg_ptr; // 定義變參起始指針
int sum=0; // 定義變參的和
int nArgValue =start; //
va_start(arg_ptr,start); // arg_ptr指向第一個變參
do
{
sum+=nArgValue; // 求和
nArgValue = va_arg(arg_ptr,int); // arg_ptr指向下一個變參
}
while(nArgValue != 0); // 判斷結束條件;結束條件是自定義為=0時結束
va_end(arg_ptr); // 復位指針
return sum;
}
函數的調用方法為Add(1,2,3,0);這樣,必須以0結尾,因為變參函數結束的判斷條件就是讀到0停止。
解釋:
所使用到的宏:
void va_start( va_list arg_ptr, prev_param );
type va_arg( va_list arg_ptr, type );
void va_end( va_list arg_ptr );
typedef char * va_list;
#define _INTSIZEOF(n) ((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define va_end(ap) ( ap = (va_list)0 )
1、首先把va_list被定義成char*,這是因為在我們目前所用的PC機上,字符指針類型可以用來存儲內存單元地址。而在有的機器上va_list是被定義成void*的
2、定義_INTSIZEOF(n)主要是為了某些需要內存的對齊的系統.這個宏的目的是為了得到最後一個固定參數的實際內存大小。在我的機器上直接用sizeof運算符來代替,對程序的運行結構也沒有影響。(後文將看到我自己的實現)。
3、va_start的定義為 &v+_INTSIZEOF(v) ,這裡&v是最後一個固定參數的起始地址,再加上其實際占用大小後,就得到了第一個可變參數的起始內存地址。所以我們運行va_start(ap, v)以後,ap指向第一個可變參數在的內存地址,有了這個地址,以後的事情就簡單了。
這裡要知道兩個事情:
⑴在intel+windows的機器上,函數棧的方向是向下的,棧頂指針的內存地址低於棧底指針,......余下全文>>