前言
今天来学习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,1,3);
d2.Init(2024,2,3);
d1.Print();
d2.Print();
return 0;
}
看完上面这段代码 有个疑问:
为什么调用的是同一个函数(不理解为什么是同一个函数可以看上一篇文章) 但打印出来一个是
2024 1 3
一个却是
2024 2 3
解释
上篇文章提到过 成员的定义是根据对象来实现的
在类的定义中 只是成员的声明 并未真实创建
那么 所有的成员函数都会多出一个指针 名为this指针 是成员函数的第一个参数
C++编译器给每个“非静态的成员函数“增加了一个隐藏的指针参数,让该指针指向当前对象(函数运行时调用该函数的对象),在函数体中所有“成员变量”的操作,都是通过该指针去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成。
所以Date类中的成员函数和函数调用其实可以理解为:
void Print(Date* const this)
{
cout <<this->_year<< "-" <<this->_month << "-"<< this->_day <<endl;
}
d1.Print(&d1);
其实和C没啥区别 只是创建者自己优化了一下 把传参这一步交给了编译器
this指针的特性
- this指针的类型:类类型*const,即成员函数中,不能给this指针赋值。
- 只能在“成员函数”的内部使用
- this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给this形参。所以对象中不存储this指针。
- this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传递,不需要用户传递
面试题
- this指针存在哪里?
this指针是形参 存在栈里面
(部分编译器用寄存器存储,如VS) - this指针可以为空吗?
// 1.下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行
class A
{
public:
void PrintA()
{
cout<<_a<<endl;
}
private:
int _a;
};
int main()
{
A* p = nullptr;
p->PrintA();
return 0;
}
p传递了一个this指针(空指针) 并且在输出_a的时候对空指针进行了解引用
所以是运行崩溃
那么留一道思考题:
// 1.下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行
class A
{
public:
void PrintA()
{
cout<<this<<endl;
}
private:
int _a;
};
int main()
{
A* p = nullptr;
(*p).PrintA();
return 0;
}
结语
关于this指针的学习到这里就结束了 我们下篇文章见~