this指针引出:
我们知道一个类可以有多个实例化对象,但是这多个实例化对象所调用的成员函数是在公共代码区;
我们先来定义一个Date类:
class Date
{
public:
void init(int year, int month, int day)
{
_year = year;
_month = month;
_day = day;
}
void print()
{
cout << _year << " " << _month << " " << _day << " " << endl;
}
private:
int _year;
int _month;
int _day;
};
int main()
{
Date d1,d2;
d1.init(2024, 5, 30);
d2.init(2024, 5, 29);
d1.print();
d2.print();
}
Date类中有init()和print()两个函数,但是这两个函数的函数体中并没有关于不同对象的区分,所以d1调用print()时,函数在怎么区分d1对象还是d2对象?------------->隐含的this指针
C++编译器给每个“非静态的成员函数“增加了一个隐藏的指针参数,让该指针指向当前对象(函数运行时调用该函数的对象),在函数体中所有“成员变量”的操作,都是通过该指针去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成。
注意:
隐含的this指针:
1、在实参和形参的位置不能显示写的,这个编译器自己加的,但是可以在类里面显示的使用:比如上面的:cout << this->_year << " " << this->_month << " " << this->_day << endl;编译时是不会报错的;
2、void print(Date* this)是写成void print(Date* const this),在这里const是修饰this的,this是不能修改的,但是它指向的内容是可以修改的;
this指针的特性:
1. this指针的类型:类类型* const,即成员函数中,不能给this指针赋值。
2. 只能在“成员函数”的内部使用
3. this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给
this形参。所以对象中不存储this指针。
4. this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传
递,不需要用户传递
面试题:
this指针存在哪里?
栈 堆 静态区 常量区 对象里面
首先排除对象里面,因为我们算sizeof的时候根本就没有算this指针的大小;
形参和局部变量都是存在栈上面的;
堆是malloc的;
静态区是存放静态数据和全局数据;
常量区是常量;
所有this指针是存放在栈上的,this指针是一个形参;
this指针可以为空吗?
1.下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行
class A
{
public:
void Print()
{
cout << "Print()" << endl;
}
private:
int _a;
};
int main()
{
A* p = nullptr;
p->Print();
return 0;
}
正确答案:C
这里:p->Print();传的是给空指针,但是在Print()函数里面并没有对这个指针进行访问,所以不会报错;
2.下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行
class A
{
public:
void PrintA()
{
cout<<_a<<endl;
}
private:
int _a;
};
int main()
{
A* p = nullptr;
p->PrintA();
return 0;
}
正确答案:B
这里cout<<_a<<endl;其实是cout<<this->_a<<endl;这里_a在this指向的空间上面,那要打印——a就要在this指向的空间上面去找,但是this指针为空,所以就会崩溃。
根据上面的例子,可知只要在成员函数内部不访问其成员变量,this指针就可以为空。
完结!!!