c++的类中,非静态的成员函数都有一个隐藏的this指针 ,在函数体中所以的成员变量都是通过this指针 来访问的。但是this指针是对用户是透明的,用户不需要显示的给this指针传参,编译时会自动接收参数。
一,this指针的注意点
- 调用成员函数时,不可以显示给this指针传参
- 定义成员函数时,不能显示的声明this指针
- 在成员函数内部,可以显示的使用this指针
用一个日期类,来讲述this指针的注意点:
class Date
{
public:
void Print()
{
cout<<_year<<"/"<<_month<<"/"<<_day<<endl;
}
Date (int year,int month,int day)
{
_year=year;
_month=month;
_day=day;
]
private:
int _year;
int _month;
int _day;
}
以上就是一个日期类了,在类里面我定义了一个成员函数Print ,这里要讲一下,在类中定义一个成员函数可能会被看成内联函数 ,想这么简单一个Print 函数一般是会别看成内联函数的。 其实这个成员函数还可以写成这样:
void Print()
{
cout<<this->_year<<"/"<<this->_month<<"/"<<this->day<<endl;
}
这样写有点画蛇添足的感觉,但是方便大家看,在成员函数中每个成员变量的访问都里离不开this指针 。这个this指针隐藏在函数形参的头一位,也就是Print(Date * this)。 比如我要调用这个成员函数:
int main()
{
Date s1(2022,2,28);
Date s2(2022,3,1);
s1.Print();
s2.Print();
}
运行结果如下: 可以看到,调用结果是符合预期的,Print函数中是利用this指针 来区分对象的。 错误用例: (1)
int main()
{
Date s1(2022,2,28);
s1.Print(&s1);
}
不可以显示的给this指针传参,虽然我们知道this指针存在,但是我们不可以给this指针传参,就当潜规则吧。
(2)
void Date::Print(Date* this);
成员函数的声明以及定义都不能有this指针,这样做不行的,可以想想设计者的初心,他是嫌弃老是传地址麻烦,所以搞了一个隐藏的this指针,你再给函数传参或者定义函数时给this指针,不是多此一举嘛。
二,this指针存在哪里
this指针是成员函数第一个隐藏的指针形参 ,一般存在栈中。但是也是由编译器决定的 ,有的就存在寄存器中。
三,this指针的面试坑题
class A
{
public:
void PrintA()
{
cout<<_a<<endl;
}
void Show()
{
cout<<"Show()"<<endl;
}
private:
int _a;
};
int main()
{
A* p = nullptr;
p->PrintA();
p->Show();
}
问:以上程序可以运行嘛?会崩溃嘛?崩溃在哪里? 这是对空指针解引用导致的内存问题,所以程序崩溃了,this指针可以为空,但是不能对空指针解引用,函数PrintA中,cout<<_a<<endl;可以写成cout<< this -> _a<<endl; 讲到这大家应该懂了。 还有一点那么Show函数有问题嘛?没问题,因为不存在对this的解引用。 我们将PrintA函数屏蔽掉,看运行结果: 没的问题。
|