调用函数时计算机常用栈来存储传递给函数的参数。栈是一种先进后出的数据结构栈有一个存储区、一个栈顶指针。栈顶指针指向堆栈中第一个可用的数据项被称为栈顶。用户可以在栈顶上方向栈中加入数据这个操作被称为压栈(Push)压栈以后栈顶自动变成新加入数据项的位置栈顶指针也随之修改。用户也可以从堆栈中取走栈顶称为弹出栈(pop)弹出栈后栈顶下的一个元素变成栈顶栈顶指针随之修改。函数调用时调用者依次把参数压栈然后调用函数函数被调用以后在堆栈中取得数据并进行计算。函数计算结束以后或者调用者、或者函数本身修改堆栈使堆栈恢复原装。在参数传递中有两个重要的问题必须要明确说明1. 当参数个数多于一个时按照什么顺序把参数压入堆栈2. 函数调用后由谁来把堆栈恢复原状。在高级语言中就是通过函数的调用方式来说明这两个问题的。常见的调用方式有stdcallcdeclfastcallthiscallthiscallnaked call下面就分别介绍这几种调用方式1. stdcallstdcall调用方式又被称为Pascal调用方式。在Microsoft C系列的C/C编译器中使用PASCAL宏WINAPI宏和CALLBACK宏来指定函数的调用方式为stdcall。stdcall调用方式的函数声明为int _stdcall function(int a, int b);stdcall的调用方式意味着1 参数从右向左一次压入堆栈2 由被调用函数自己来恢复堆栈3 函数名自动加前导下划线后面紧跟着一个,其后紧跟着参数的尺寸上面那个函数翻译成汇编语言将变成push b 先压入第二个参数push a 再压入第一个参数call function 调用函数在编译时此函数的名字被翻译为_function82. cdeclcdecl调用方式又称为C调用方式是C语言缺省的调用方式它的语法为int function(int a, int b) // 不加修饰符就是C调用方式int _cdecl function(int a, int b) // 明确指定用C调用方式cdecl的调用方式决定了1 参数从右向左依次压入堆栈2 由调用者恢复堆栈3 函数名自动加前导下划线由于是由调用者来恢复堆栈因此C调用方式允许函数的参数个数是不固定的这是C语言的一大特色。此方式的函数被翻译为push b // 先压入第二个参数push a // 在压入第一个参数call funtion // 调用函数add esp, 8 // 清理堆栈在编译时此方式的函数被翻译成_function3. fastcallfastcall 按照名字上理解就可以知道它是一种快速调用方式。此方式的函数的第一个和第二个DWORD参数通过ecx和edx传递后面的参数从右向左的顺序压入栈。被调用函数清理堆栈。函数名修个规则同stdcall其声明语法为int fastcall function(int a, int b);4. thiscallthiscall 调用方式是唯一一种不能显示指定的修饰符。它是c类成员函数缺省的调用方式。由于成员函数调用还有一个this指针因此必须用这种特殊的调用方式。thiscall调用方式意味着参数从右向左压入栈。如果参数个数确定this指针通过ecx传递给被调用者如果参数个数不确定this指针在所有参数压入栈后被压入栈。参数个数不定的由调用者清理堆栈否则由函数自己清理堆栈。可以看到对于参数个数固定的情况它类似于stdcall不定时则类似于cdecl。5. naked call是一种比较少见的调用方式一般高级程序设计语言中不常见。函数的声明调用方式和实际调用方式必须一致必然编译器会产生混乱。函数名字修改规则1. C编译时函数名修饰约定规则__stdcall调用约定在输出函数名前加上一个下划线前缀后面加上一个“”符号和其参数的字节数格式为_function8。__cdecl调用约定仅在输出函数名前加上一个下划线前缀格式为_function。__fastcall调用约定在输出函数名前加上一个“”符号后面也是一个“”符号和其参数的字节数格式为function8。它们均不改变输出函数名中的字符大小写这和PASCAL调用约定不同PASCAL约定输出的函数名无任何修饰且全部大写。2. C编译时函数名修饰约定规则__stdcall调用约定1以“?”标识函数名的开始后跟函数名2函数名后面以“YG”标识参数表的开始后跟参数表3参数表以代号表示X--void D--charE--unsigned charF--shortH--intI--unsigned intJ--longK--unsigned longM--floatN--double_N--bool....PA--表示指针后面的代号表明指针类型如果相同类型的指针连续出现以“0”代替一个“0”代表一次重复4参数表的第一项为该函数的返回值类型其后依次为参数的数据类型,指针标识在其所指数据类型前5参数表后以“Z”标识整个名字的结束如果该函数无参数则以“Z”标识结束。其格式为“?functionnameYG*****Z”或“?functionnameYG*XZ”例如int Test1char *var1,unsigned long-----“?Test1YGHPADKZ”void Test2 -----“?Test2YGXXZ”__cdecl调用约定规则同上面的_stdcall调用约定只是参数表的开始标识由上面的“YG”变为“YA”。
C/C++函数调用的几种方式总结
调用函数时计算机常用栈来存储传递给函数的参数。栈是一种先进后出的数据结构栈有一个存储区、一个栈顶指针。栈顶指针指向堆栈中第一个可用的数据项被称为栈顶。用户可以在栈顶上方向栈中加入数据这个操作被称为压栈(Push)压栈以后栈顶自动变成新加入数据项的位置栈顶指针也随之修改。用户也可以从堆栈中取走栈顶称为弹出栈(pop)弹出栈后栈顶下的一个元素变成栈顶栈顶指针随之修改。函数调用时调用者依次把参数压栈然后调用函数函数被调用以后在堆栈中取得数据并进行计算。函数计算结束以后或者调用者、或者函数本身修改堆栈使堆栈恢复原装。在参数传递中有两个重要的问题必须要明确说明1. 当参数个数多于一个时按照什么顺序把参数压入堆栈2. 函数调用后由谁来把堆栈恢复原状。在高级语言中就是通过函数的调用方式来说明这两个问题的。常见的调用方式有stdcallcdeclfastcallthiscallthiscallnaked call下面就分别介绍这几种调用方式1. stdcallstdcall调用方式又被称为Pascal调用方式。在Microsoft C系列的C/C编译器中使用PASCAL宏WINAPI宏和CALLBACK宏来指定函数的调用方式为stdcall。stdcall调用方式的函数声明为int _stdcall function(int a, int b);stdcall的调用方式意味着1 参数从右向左一次压入堆栈2 由被调用函数自己来恢复堆栈3 函数名自动加前导下划线后面紧跟着一个,其后紧跟着参数的尺寸上面那个函数翻译成汇编语言将变成push b 先压入第二个参数push a 再压入第一个参数call function 调用函数在编译时此函数的名字被翻译为_function82. cdeclcdecl调用方式又称为C调用方式是C语言缺省的调用方式它的语法为int function(int a, int b) // 不加修饰符就是C调用方式int _cdecl function(int a, int b) // 明确指定用C调用方式cdecl的调用方式决定了1 参数从右向左依次压入堆栈2 由调用者恢复堆栈3 函数名自动加前导下划线由于是由调用者来恢复堆栈因此C调用方式允许函数的参数个数是不固定的这是C语言的一大特色。此方式的函数被翻译为push b // 先压入第二个参数push a // 在压入第一个参数call funtion // 调用函数add esp, 8 // 清理堆栈在编译时此方式的函数被翻译成_function3. fastcallfastcall 按照名字上理解就可以知道它是一种快速调用方式。此方式的函数的第一个和第二个DWORD参数通过ecx和edx传递后面的参数从右向左的顺序压入栈。被调用函数清理堆栈。函数名修个规则同stdcall其声明语法为int fastcall function(int a, int b);4. thiscallthiscall 调用方式是唯一一种不能显示指定的修饰符。它是c类成员函数缺省的调用方式。由于成员函数调用还有一个this指针因此必须用这种特殊的调用方式。thiscall调用方式意味着参数从右向左压入栈。如果参数个数确定this指针通过ecx传递给被调用者如果参数个数不确定this指针在所有参数压入栈后被压入栈。参数个数不定的由调用者清理堆栈否则由函数自己清理堆栈。可以看到对于参数个数固定的情况它类似于stdcall不定时则类似于cdecl。5. naked call是一种比较少见的调用方式一般高级程序设计语言中不常见。函数的声明调用方式和实际调用方式必须一致必然编译器会产生混乱。函数名字修改规则1. C编译时函数名修饰约定规则__stdcall调用约定在输出函数名前加上一个下划线前缀后面加上一个“”符号和其参数的字节数格式为_function8。__cdecl调用约定仅在输出函数名前加上一个下划线前缀格式为_function。__fastcall调用约定在输出函数名前加上一个“”符号后面也是一个“”符号和其参数的字节数格式为function8。它们均不改变输出函数名中的字符大小写这和PASCAL调用约定不同PASCAL约定输出的函数名无任何修饰且全部大写。2. C编译时函数名修饰约定规则__stdcall调用约定1以“?”标识函数名的开始后跟函数名2函数名后面以“YG”标识参数表的开始后跟参数表3参数表以代号表示X--void D--charE--unsigned charF--shortH--intI--unsigned intJ--longK--unsigned longM--floatN--double_N--bool....PA--表示指针后面的代号表明指针类型如果相同类型的指针连续出现以“0”代替一个“0”代表一次重复4参数表的第一项为该函数的返回值类型其后依次为参数的数据类型,指针标识在其所指数据类型前5参数表后以“Z”标识整个名字的结束如果该函数无参数则以“Z”标识结束。其格式为“?functionnameYG*****Z”或“?functionnameYG*XZ”例如int Test1char *var1,unsigned long-----“?Test1YGHPADKZ”void Test2 -----“?Test2YGXXZ”__cdecl调用约定规则同上面的_stdcall调用约定只是参数表的开始标识由上面的“YG”变为“YA”。