C/C++函數(shù)調(diào)用的幾種方式總結(jié)
調(diào)用函數(shù)時(shí),計(jì)算機(jī)常用棧來存儲(chǔ)傳遞給函數(shù)的參數(shù)。
棧是一種先進(jìn)后出的數(shù)據(jù)結(jié)構(gòu),棧有一個(gè)存儲(chǔ)區(qū)、一個(gè)棧頂指針。棧頂指針指向堆棧中第一個(gè)可用的數(shù)據(jù)項(xiàng)(被稱為棧頂)。用戶可以在棧頂上方向棧中加入數(shù)據(jù),這個(gè)操作被稱為壓棧(Push),壓棧以后,棧頂自動(dòng)變成新加入數(shù)據(jù)項(xiàng)的位置,棧頂指針也隨之修改。用戶也可以從堆棧中取走棧頂,稱為彈出棧(pop),彈出棧后,棧頂下的一個(gè)元素變成棧頂,棧頂指針隨之修改。函數(shù)調(diào)用時(shí),調(diào)用者依次把參數(shù)壓棧,然后調(diào)用函數(shù),函數(shù)被調(diào)用以后,在堆棧中取得數(shù)據(jù),并進(jìn)行計(jì)算。函數(shù)計(jì)算結(jié)束以后,或者調(diào)用者、或者函數(shù)本身修改堆棧,使堆?;謴?fù)原裝。
在參數(shù)傳遞中,有兩個(gè)重要的問題必須要明確說明:
1. 當(dāng)參數(shù)個(gè)數(shù)多于一個(gè)時(shí),按照什么順序把參數(shù)壓入堆棧;
2. 函數(shù)調(diào)用后,由誰來把堆?;謴?fù)原狀。
在高級(jí)語言中,就是通過函數(shù)的調(diào)用方式來說明這兩個(gè)問題的。常見的調(diào)用方式有:
stdcall
cdecl
fastcall
thiscall
thiscall
naked call
下面就分別介紹這幾種調(diào)用方式:
1. stdcall
stdcall調(diào)用方式又被稱為Pascal調(diào)用方式。在Microsoft C++系列的C/C++編譯器中,使用PASCAL宏,WINAPI宏和CALLBACK宏來指定函數(shù)的調(diào)用方式為stdcall。
stdcall調(diào)用方式的函數(shù)聲明為:
int _stdcall function(int a, int b);
stdcall的調(diào)用方式意味著:
(1) 參數(shù)從右向左一次壓入堆棧
(2) 由被調(diào)用函數(shù)自己來恢復(fù)堆棧
(3) 函數(shù)名自動(dòng)加前導(dǎo)下劃線,后面緊跟著一個(gè)@,其后緊跟著參數(shù)的尺寸
上面那個(gè)函數(shù)翻譯成匯編語言將變成:
push b 先壓入第二個(gè)參數(shù)
push a 再壓入第一個(gè)參數(shù)
call function 調(diào)用函數(shù)
在編譯時(shí),此函數(shù)的名字被翻譯為_function@8
2. cdecl
cdecl調(diào)用方式又稱為C調(diào)用方式,是C語言缺省的調(diào)用方式,它的語法為:
int function(int a, int b) // 不加修飾符就是C調(diào)用方式
int _cdecl function(int a, int b) // 明確指定用C調(diào)用方式
cdecl的調(diào)用方式?jīng)Q定了:
(1) 參數(shù)從右向左依次壓入堆棧
(2) 由調(diào)用者恢復(fù)堆棧
(3) 函數(shù)名自動(dòng)加前導(dǎo)下劃線
由于是由調(diào)用者來恢復(fù)堆棧,因此C調(diào)用方式允許函數(shù)的參數(shù)個(gè)數(shù)是不固定的,這是C語言的一大特色。
此方式的函數(shù)被翻譯為:
push b // 先壓入第二個(gè)參數(shù)
push a // 在壓入第一個(gè)參數(shù)
call funtion // 調(diào)用函數(shù)
add esp, 8 // 清理堆棧
在編譯時(shí),此方式的函數(shù)被翻譯成:_function
3. fastcall
fastcall 按照名字上理解就可以知道,它是一種快速調(diào)用方式。此方式的函數(shù)的第一個(gè)和第二個(gè)DWORD參數(shù)通過ecx和edx傳遞,
后面的參數(shù)從右向左的順序壓入棧。
被調(diào)用函數(shù)清理堆棧。
函數(shù)名修個(gè)規(guī)則同stdcall
其聲明語法為:
int fastcall function(int a, int b);
4. thiscall
thiscall 調(diào)用方式是唯一一種不能顯示指定的修飾符。它是c++類成員函數(shù)缺省的調(diào)用方式。由于成員函數(shù)調(diào)用還有一個(gè)this指針,因此必須用這種特殊的調(diào)用方式。
thiscall調(diào)用方式意味著:
參數(shù)從右向左壓入棧。
如果參數(shù)個(gè)數(shù)確定,this指針通過ecx傳遞給被調(diào)用者;如果參數(shù)個(gè)數(shù)不確定,this指針在所有參數(shù)壓入棧后被壓入棧。
參數(shù)個(gè)數(shù)不定的,由調(diào)用者清理堆棧,否則由函數(shù)自己清理堆棧。
可以看到,對(duì)于參數(shù)個(gè)數(shù)固定的情況,它類似于stdcall,不定時(shí)則類似于cdecl。
5. naked call
是一種比較少見的調(diào)用方式,一般高級(jí)程序設(shè)計(jì)語言中不常見。
函數(shù)的聲明調(diào)用方式和實(shí)際調(diào)用方式必須一致,必然編譯器會(huì)產(chǎn)生混亂。
函數(shù)名字修改規(guī)則:
1. C編譯時(shí)函數(shù)名修飾約定規(guī)則:
__stdcall調(diào)用約定在輸出函數(shù)名前加上一個(gè)下劃線前綴,后面加上一個(gè)“@”符號(hào)和其參數(shù)的字節(jié)數(shù),格式為_function@8。
__cdecl調(diào)用約定僅在輸出函數(shù)名前加上一個(gè)下劃線前綴,格式為_function。
__fastcall調(diào)用約定在輸出函數(shù)名前加上一個(gè)“@”符號(hào),后面也是一個(gè)“@”符號(hào)和其參數(shù)的字節(jié)數(shù),格式為@function@8。
它們均不改變輸出函數(shù)名中的字符大小寫,這和PASCAL調(diào)用約定不同,PASCAL約定輸出的函數(shù)名無任何修飾且全部大寫。
2. C++編譯時(shí)函數(shù)名修飾約定規(guī)則:
__stdcall調(diào)用約定:
(1)以“?”標(biāo)識(shí)函數(shù)名的開始,后跟函數(shù)名;
(2)函數(shù)名后面以“@@YG”標(biāo)識(shí)參數(shù)表的開始,后跟參數(shù)表;
(3)參數(shù)表以代號(hào)表示:
X--void ,
D--char,
E--unsigned char,
F--short,
H--int,
I--unsigned int,
J--long,
K--unsigned long,
M--float,
N--double,
_N--bool,
....
PA--表示指針,后面的代號(hào)表明指針類型,如果相同類型的指針連續(xù)出現(xiàn),以“0”代替,一個(gè)“0”代
表一次重復(fù);
(4)參數(shù)表的第一項(xiàng)為該函數(shù)的返回值類型,其后依次為參數(shù)的數(shù)據(jù)類型,指針標(biāo)識(shí)在其所指數(shù)據(jù)類型前;
(5)參數(shù)表后以“@Z”標(biāo)識(shí)整個(gè)名字的結(jié)束,如果該函數(shù)無參數(shù),則以“Z”標(biāo)識(shí)結(jié)束。
其格式為“?functionname@@YG*****@Z”或“?functionname@@YG*XZ”,例如
int Test1(char *var1,unsigned long)-----“?Test1@@YGHPADK@Z”
void Test2() -----“?Test2@@YGXXZ”
__cdecl調(diào)用約定:
規(guī)則同上面的_stdcall調(diào)用約定,只是參數(shù)表的開始標(biāo)識(shí)由上面的“@@YG”變?yōu)椤癅@YA”。
__fastcall調(diào)用約定:
規(guī)則同上面的_stdcall調(diào)用約定,只是參數(shù)表的開始標(biāo)識(shí)由上面的“@@YG”變?yōu)椤癅@YI”。
VC++對(duì)函數(shù)的省缺聲明是"__cedcl",將只能被C/C++調(diào)用。
上一篇:淺析直接插入排序與折半插入排序
欄 目:C語言
下一篇:純C語言:分治假幣問題源碼分享
本文標(biāo)題:C/C++函數(shù)調(diào)用的幾種方式總結(jié)
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/3842.html
您可能感興趣的文章
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用函數(shù)刪除字符
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)式函數(shù)庫(kù)
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)數(shù)怎么表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段函數(shù)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排序法函數(shù)
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段函數(shù)
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求階乘


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10delphi制作wav文件的方法
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 04-02jquery與jsp,用jquery
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?