在C/C++中,對函數(shù)參數(shù)的掃描是從后向前的。C/C++的函數(shù)參數(shù)是通過壓入堆棧的方式來給函數(shù)傳參數(shù)的(堆棧是一種先進后出的數(shù)據(jù)結(jié)構(gòu)),最先壓入的參數(shù)最后出來,在計算機的內(nèi)存中,數(shù)據(jù)有2塊,一塊是堆,一塊是棧(函數(shù)參數(shù)及局部變量在這里),而棧是從內(nèi)存的高地址向低地址生長的,控制生長的就是堆棧指針了,最先壓入的參數(shù)是在最上面,就是說在所有參數(shù)的最后面,最后壓入的參數(shù)在最下面,結(jié)構(gòu)上看起來是第一個,所以最后壓入的參數(shù)總是能夠被函數(shù)找到,因為它就在堆棧指針的上方。printf的第一個被找到的參數(shù)就是那個字符指針,就是被雙引號括起來的那一部分,函數(shù)通過判斷字符串里控制參數(shù)的個數(shù)來判斷參數(shù)個數(shù)及數(shù)據(jù)類型,通過這些就可算出數(shù)據(jù)需要的堆棧指針的偏移量了,下面給出printf("%d,%d",a,b);(其中a、b都是int型的)的匯編代碼.
10年積累的網(wǎng)站建設(shè)、網(wǎng)站制作經(jīng)驗,可以快速應(yīng)對客戶對網(wǎng)站的新想法和需求。提供各種問題對應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識你,你也不認(rèn)識我。但先網(wǎng)站設(shè)計制作后付款的網(wǎng)站建設(shè)流程,更有堆龍德慶免費網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
.section
.data
string out = "%d,%d"
push b //最后的先壓入棧中
push a //最先的后壓入棧中
push $out//參數(shù)控制的那個字符串常量是最后被壓入的
call printf
你會看到,參數(shù)是最后的先壓入棧中,最先的后壓入棧中,參數(shù)控制的那個字符串常量是最后被壓入的,所以這個常量總是能被找到的。
通常情況下函數(shù)可變參數(shù)表的長度是已知的,通過num參數(shù)傳入,這種函數(shù)比較容易實現(xiàn)。
當(dāng)然可以,最典型的就是printf函數(shù)了,這個函數(shù)采用的就是可以改變形參數(shù)量的方法,百度上搜索:“C語言不定參數(shù)”就可以查到相關(guān)信息了C語言中的不定參數(shù)
#include?stdarg.h??????????//可變參數(shù)函數(shù)必要的頭文件
#include?stdio.h
long?int?sum(unsigned?int?argc,...)????//可變參數(shù)函數(shù)必須包含至少一個確定的參數(shù)
{
long?int?srt=0;
int?va;
unsigned?int?i;
va_list?vl;????????????????????????//用于指向可變參數(shù)列表
va_start(vl,argc);?????????????????//初始化,va_start()調(diào)用的第一個參數(shù)是所編寫的可變參數(shù)函數(shù)的最后一個確定的參數(shù)
for?(i?=?0;?iargc;?i++)?{
srt+=(int)va_arg(vl,int);??????//從參數(shù)列表按提取一個int類型的值參與相加運算
}
va_end(vl);????????????????????????//清理
return?srt;
}
int?main(int?argc,?char*?argv[])
{
printf("%ld\n",sum(3,1,2,3));
return?0;
}
定義如下:
函數(shù)返回值 函數(shù)名(形參1, 形參2, ...)
舉例說明:
int??func(int?a,?int?b,?...)
{
//函數(shù)體
......
}
printf()函數(shù)就是一個參數(shù)可變的函數(shù),其函數(shù)原型為:
int?printf?(const?char?*__format,?...);