1. this
是一个指针,里面放置的是当前对象的地址(成员函数执行时,调用该成员函数的对象)
this指针是类“成员函数”第一个隐藏的参数,该指针指向调用成员函数的对象(当前对象)
2. this指针的特性
(1) 只能在成员函数中使用
(2) this指针的类型:类类型* const
(3) this指针没有存储在对象中,因此不会影响对象的大小,而是在成员函数运行时,时时刻刻指向当前对象
(4) this指针是“成员函数”第一个隐藏的参数,“隐藏的”—用户在实现成员函数时,不用显式给出,该参数是编译器自动添加的也是编译器自动来进行传递的
(5) this指针主要是通过ecx寄存器来传递的
3. this指针为什么主要是通过ecx寄存器传递的呢?
- example
#include <iostream>
using namespace std;
class Student
{
public:
char _name[20];
char _gender[3];
int _age;
public:
void InitStudent(char name[], char gender[], int age)
{
strcpy_s(this->_name, name);
strcpy_s(this->_gender, gender);
this->_age = age;
}
void PrintStudent()
{
cout << this << endl;
cout << _name << "-" << _gender << "-" << _age << endl;
}
void SetAge(int age)
{
cout << this << endl;
_age = age;
}
void TestFunc(...)
{
}
};
int main()
{
Student s1, s2, s3;
s1.InitStudent("Peter", "男", 18);
s2.InitStudent("David", "男", 19);
s3.InitStudent("Lily", "女", 18);
s1.PrintStudent();
s2.PrintStudent();
s3.PrintStudent();
s1.TestFunc(1);
s2.TestFunc(1, 2);
s3.TestFunc(1, 2, 3);
return 0;
}
this 指针主要是通过ecx寄存器来传递的,但在这个例子中,this指针是通过压栈来传递的
当我们只给InitStudent()和TestFunc()声明的时候,我们可以看到会报如下错误
4. 函数的调用约定
从右到左依次入栈:__stdcall,__cdecl,__thiscall,__fastcall
从左到右依次入栈:__pascal
- (1) __stdcall
__stdcall是StandardCall的缩写,是C++的标准调用方式。__stdcall调用约定的规则如下:
A、所有参数从右到左依次入栈,如果是调用类成员的话,最后一个入栈的是this指针。
B、被调用函数自动清理堆栈,返回值在EAX。
C、函数修饰名约定:VC将函数编译后会在函数名前面加上下划线前缀,在函数名后加上"@"和参数的字节数。
- (2) __cdecl
__cdecl是C DECLaration的缩写(declaration,声明),表示C语言默认的函数调用方法。__cdecl调用约定规则如下:
A、所有参数从右到左依次入栈
B、所有参数由调用者清除,称为手动清栈。返回值在EAX中
C、函数修饰名约定:VC将函数编译后会在函数名前面加上下划线前缀
由于由调用者清理栈,所以允许可变参数函数存在,如int sprintf(char buffer,const char format,…)。
- (3) __fastcall
__fastcall是快速调用约定,通过寄存器来传送参数。__fastcall调用约定的规则如下:
A、用ECX和EDX传送前两个双字(DWORD)或更小的参数,剩下的参数仍旧自右向左压栈传送
B、被调用函数在返回前清理传送参数的内存栈 ,返回值在EAX中
C、函数修饰名约定:VC将函数编译后会在函数名前面加上"@“前缀,在函数名后加上”@"和参数的字节数 。
- (4) __thiscall
__thiscall是唯一一个不能明确指明的函数修饰符,thiscall只能用于C++类成员函数的调用,同时__thiscall也是C++成员函数缺省的调用约定。由于成员函数调用还有一个this指针,因此必须特殊处理。
thiscall调用约定如下:
A、采用桟传递参数,参数从右向左入栈。如果参数个数确定,this指针通过ECX传递给被调用者;如果参数个数不确定,this指针在所有参数压栈后被压入堆栈。
B、对参数个数不定的,调用者清理堆栈,否则由被调函数清理堆栈
__thiscall 不是关键字,程序员不能使用。
- (5) __pascal
__pascal 语言的调用约定,跟 __stdcall 一样,参数按照从左至右的方式入栈,函数自身清理堆栈,返回值在EAX中。VC 中已经废弃,建议使用 stdcall 代替。