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. 函数的调用约定
(1) __stdcall从右到左依次入栈:__stdcall,__cdecl,__thiscall,__fastcall
从左到右依次入栈:__pascal
(2) __cdecl__stdcall是StandardCall的缩写,是C++的标准调用方式。__stdcall调用约定的规则如下:
A、所有参数从右到左依次入栈,如果是调用类成员的话,最后一个入栈的是this指针。
B、被调用函数自动清理堆栈,返回值在EAX。
C、函数修饰名约定:VC将函数编译后会在函数名前面加上下划线前缀,在函数名后加上"@"和参数的字节数。
(3) __fastcall__cdecl是C DECLaration的缩写(declaration,声明),表示C语言默认的函数调用方法。__cdecl调用约定规则如下:
A、所有参数从右到左依次入栈
B、所有参数由调用者清除,称为手动清栈。返回值在EAX中
C、函数修饰名约定:VC将函数编译后会在函数名前面加上下划线前缀
由于由调用者清理栈,所以允许可变参数函数存在,如int sprintf(char buffer,const char format,…)。
(4) __thiscall__fastcall是快速调用约定,通过寄存器来传送参数。__fastcall调用约定的规则如下:
A、用ECX和EDX传送前两个双字(DWORD)或更小的参数,剩下的参数仍旧自右向左压栈传送
B、被调用函数在返回前清理传送参数的内存栈 ,返回值在EAX中
C、函数修饰名约定:VC将函数编译后会在函数名前面加上"@“前缀,在函数名后加上”@"和参数的字节数 。
(5) __pascal__thiscall是唯一一个不能明确指明的函数修饰符,thiscall只能用于C++类成员函数的调用,同时__thiscall也是C++成员函数缺省的调用约定。由于成员函数调用还有一个this指针,因此必须特殊处理。
thiscall调用约定如下:
A、采用桟传递参数,参数从右向左入栈。如果参数个数确定,this指针通过ECX传递给被调用者;如果参数个数不确定,this指针在所有参数压栈后被压入堆栈。
B、对参数个数不定的,调用者清理堆栈,否则由被调函数清理堆栈
__thiscall 不是关键字,程序员不能使用。
__pascal 语言的调用约定,跟 __stdcall 一样,参数按照从左至右的方式入栈,函数自身清理堆栈,返回值在EAX中。VC 中已经废弃,建议使用 stdcall 代替。