函数符号(函数的名字修饰):在编译阶段生成的字符串,来指明函数的定义或原型 。
C函数符号生成规则:
_cdel调用约定(C标准调用约定):函数名前加下划线。
_stdcall调用约定:函数名前加下划线,函数名后加“@”符号和其参数字节。
_fastcall调用约定:函数名前加“@”符号,函数名后加“@”字符和其参数字节。
C++函数符号生成规则:
_cdecl调用约定:“?”+函数名+参数表的开始标识 “@@YA” + 函数返回类型代号+参数类型代号 +结束标识“@Z”或“Z”(无参数)。
_stdcall调用约定:“?”+函数名+参数表的开始标识“@@YG”+函数返回类型代号+参数类型代号 +结束标识“@Z”或“Z”(无参数)。
_fastcal调用约定:“?”+函数名+参数表的开始标识 “@@YI”+ 函数返回类型代号+参数类型代号 +结束标识“@Z”或“Z”(无参数)。
X | D | E | F | H | I | J | K | M | N | _N | U | PA+ | PB+ | ... |
void | char | unsigned char | short | int | unsigned int | long | unsigned long | float | double | bool | struct | 指针+其类型 | const指针+其类型 | ... |
_thiscall调用约定(类成员方法的约定):“?” +函数名+ “@”字符引导的类名+参数表的开始标识(函数性质决定)+ 函数返回类型代号+参数类型代号 +结束标识“@Z”或“Z”(无参数)。
参数表的开始标识:公有(public)成员函数的标识“@@QAE”,保护(protected)成员函数的标识“@@IAE”,私有(private)成员函数“@@AAE”;如果使用Const ,则对应的标识“@@QBE”,“@@IBE”,“@@ABE”。
形参内存的开辟和清理:
_cdecl调用约定:调用方开辟和清理;
_stdcall调用约定:调用方开辟,被调用方清理;
_fastcal调用约定:前两个形参是由寄存器带入(被调用方没有开辟内存),以后的形参和_stdcall调用约定相同
C++函数重载:
在 同作用域 下定义 相同函数名,不同参数列表 函数。
注意:仅仅函数返回值类型不同或仅仅默认实参不同都不能构成函数重载 ,因为调用时,编译器只能通过实参的类型确定应该调用哪一个函数。
因为编译器只使用形参列表来区分重载的函数,所以仅仅返回值不同和 默认实参不同,编译无法区分其函数,然后报错。
int Max(int a, int b); //?Max@@YAHHH@Z
double Max(int a, int b); //?Max@@YANHH@Z
double Max1(int a, int b); //?Max1@@YANHH@Z
double Max1(int a, double b); //?Max1@@YANHN@Z
当调用一个重载函数时,C++编译器通过函数符号(函数修饰名)来选择恰当的函数。
int Max(int a, int b)//?Max@@YAHHH@Z
{
return a > b ? a : b;
}
double Max(double a, double b)//?Max@@YANNN@Z
{
return a > b ? a : b;
}
char Max(char a, char b)
{
return a > b ? a : b;
}
int main()
{
int max = Max(10, 20);
char max = Max('a', 'b');
double max = Max(0.23,0.21);
return 0;
}