函数调用协议

当参数个数多于一个时,按照什么顺序把参数压入堆栈函数调用后,由谁来把堆栈恢复原装。在高级语言中,通过函数调用约定来说明这两个问题。常见的调用约定有

  : stdcallcdeclfastcallthiscallnaked call


(1)__stdcall详解 --Windows API默认的函数调用协议。

stdcall调用约定声明的语法为(以前文的那个函数为例): int __stdcall function(int a,int b)
stdcall的调用约定意味着:1参数从右向左压入堆栈2)函数自身修改堆栈 3)函数名自动加前导的下划线,后面紧跟一个@符号,其后紧跟着参数的尺寸。
以上述这个函数为例,参数b首先被压栈,然后是参数a,函数调用function(1,2)调用处
而在编译时,这个函数的名字被翻译成_function@8

(2)cdecl调用约定: --C/C++默认的函数调用协议。 
cdecl调用约定又称为C调用约定,是C语言缺省的调用约定,它的定义语法是:
int function (int a ,int b) //不加修饰就是C调用约定
int function (int a ,int b) //不加修饰就是C调用约定
int __cdecl function(int a,int b)//明确指出C调用约定
在写本文时,出乎我的意料,发现cdecl调用约定的参数压栈顺序是和stdcall是一样的,参数首先由右向左压入堆栈。所不同的是,函数本身不清理堆栈,调用者负责清理堆栈。由于这种变化,C调用约定允许函数的参数的个数是不固定的,这也是C语言的一大特色。对于前面的function函数,使用cdecl后的汇编码变成:

(3)fastcall --适用于对性能要求较高的场合。
fastcall调用约定和stdcall类似,它意味着:从左开始不大于4字节的参数放入CPU的ECX和EDX寄存器,其余参数从右向左入栈。
函数的第一个和第二个DWORD参数(或者尺寸更小的)通过ecxedx传递,其他参数通过从右向左的顺序压栈被调用函数清理堆栈函数名修改规则同stdcall
其声明语法为:int fastcall function(int a,int b)
(4)thiscall
thiscall是唯一一个不能明确指明的函数修饰,因为thiscall不是关键字。它是C++类成员函数缺省的调用约定。由于成员函数调用还有一个this指针,因此必须特殊处理,thiscall意味着:
参数从右向左入栈如果参数个数确定,this指针通过ecx传递给被调用者;如果参数个数不确定,this指针在所有参数压栈后被压入堆栈。
对参数个数不定的,调用者清理堆栈,否则函数自己清理堆栈




{

__stdcall、__cdcel和__fastcall三者的区别

一、概述

__stdcall、__cdecl和__fastcall是三种函数调用协议,函数调用协议会影响函数参数的入栈方式、栈内数据的清除方式、编译器函数名的修饰规则等。

 

二、调用协议常用场合 

__stdcall:Windows API默认的函数调用协议。

 __cdecl:C/C++默认的函数调用协议。 

__fastcall:适用于对性能要求较高的场合。

 

三、 函数参数入栈方式 

__stdcall:函数参数由右向左入栈。

 __cdecl:函数参数由右向左入栈。

 __fastcall:从左开始不大于4字节的参数放入CPU的ECX和EDX寄存器,其余参数从右向左入栈。

 

四、栈内数据清除方式 

__stdcall:函数调用结束后由被调用函数清除栈内数据。 

__cdecl:函数调用结束后由函数调用者清除栈内数据。 

__fastcall:函数调用结束后由被调用函数清除栈内数据。 

 

五、常见问题 

  1. __fastcall在寄存器中放入不大于4字节的参数,故性能较高,适用于需要高性能的场合。 
  2. 不同编译器设定的栈结构不尽相同,跨开发平台时由函数调用者清除栈内数据不可行。 
  3. 某些函数的参数是可变的,如printf函数,这样的函数只能由函数调用者清除栈内数据。
  4. 由调用者清除栈内数据时,每次调用都包含清除栈内数据的代码,故可执行文件较大。 

 

六、C语言编译器函数名称修饰规则 

__stdcall:编译后,函数名被修饰为"_functionname@number"。 

__cdecl:编译后,函数名被修饰为"_functionname"。 

__fastcall:编译后, 函数名给修饰为"@functionname@nmuber"。 

注:"functionname"为函数名,"number"为参数字节数。 

注:函数实现和函数定义时如果使用了不同的函数调用协议,则无法实现函数调用。 

 

七、C++语言编译器函数名称修饰规则 

__stdcall:编译后,函数名被修饰为"?functionname@@YG******@Z"。 

__cdecl:编译后,函数名被修饰为"?functionname@@YA******@Z"。 

__fastcall:编译后,函数名被修饰为"?functionname@@YI******@Z"。 

注:"******"为函数返回值类型和参数类型表。

 注:函数实现和函数定义时如果使用了不同的函数调用协议,则无法实现函数调用。 C语言和C++语言间如果不进行特殊处理,也无法实现函数的互相调用。


}
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值