C++基础——this指针

this指针详解

一、问题

1.一个类中的不同对象在调用自己的成员函数时,其实它们调用的是同一段函数代码,那么成员函数如何知道要访问哪个对象的数据成员呢?

没错,就是通过this指针。每个对象都拥有一个this指针,this指针记录对象的内存地址,当我们调用成员函数时,成员函数默认第一个参数为T* const register this,大多数编译器通过ecx寄存器传递this指针,通过 this 这个隐式参数可以访问该对象的数据成员。

2.类的成员函数为什么不能用static和const同时修饰?

类中用const修饰的函数通常用来防止修改对象的数据成员,函数末尾的const是用来修饰this指针,防止在函数内对数据成员进行修改,而静态函数中是没有this指针的,无法访问到对象的数据成员,与C++ static语义冲突,所以不能。

二、this指针注意点

1.C++中this关键字是一个指向对象自己的一个常量指针,不能给this赋值;
2.只有成员函数才有this指针,友元函数不是类的成员函数,没有this指针;
3.同样静态函数也是没有this指针的,静态函数如同静态变量一样,不属于具体的哪一个对象;
4.this指针作用域在类成员函数内部,在类外也无法获取;
5.this指针并不是对象的一部分,this指针所占的内存大小是不会反应在sizeof操作符上的。

三、this指针的使用

1.在类的非静态成员函数中返回类对象本身的时候,直接使用 return *this,比如类的默认取址运算符重载函数,另外,也可以返回*this的引用,这样可以像输入输出流那样进行“级联”操作;
2.修改类成员变量或参数与成员变量名相同时,如this->a = a (写成a = a编译不过);
3.在class定义时要用到类型变量自身时,因为这时候还不知道变量名,就用this这样的指针来使用变量自身。

四、this指针探讨

1.this指针是什么时候创建的?

对象new的过程中创建的,具体哪个阶段有待进一步深入了解。

2. this指针存放在何处?

this指针会因编译器不同而有不同的放置位置。可能是栈,也可能是寄存器,甚至全局变量。在汇编级别里面,一个值只会以3种形式出现:立即数、寄存器值和内存变量值。不是存放在寄存器就是存放在内存中,它们并不是和高级语言变量对应的。

3.为什么C++ NULL对象指针可以调用非虚成员函数,而Java中却不行?

C++语言是静态绑定的,这也是C++语言和Java语言的一个显著区别。类的成员函数并不与特定对象绑定,所有成员函数共用一份成员函数体,当程序编译后,成员函数的地址即已经确定。另外,C++只关心你的指针类型,不关心指针指向的对象是否有效,C++要求程序员自己保证指针的有效性。况且在有些系统上,地址0也是有效的,理论上完全可以构造一个在地址0上的对象,所以C++中nullptr对象调用成员函数并无不可 。

nullptr对象调用成员函数时,只要不访问此对象独有的内存部分,则程序正常运行,因为不会使用this,一旦访问此对象的成员变量,则程序崩溃。当然nullptr调用虚方法是不能正常运行的(虚函数有虚表,会占用内存空间),虚方法调用是依赖于this指针的。可以这样理解,你给函数传递了错误的参数,但在该函数内部并没有使用该参数,所以其不影响函数的运行。可以参考下面代码:

 1 #include <iostream>
 2 using namespace std;
 3 
 4 class CPeople
 5 {
 6 public:
 7     CPeople(const std::string& name, int age)
 8       : mName(name), mAge(age){}
 9      ~CPeople();
10  
11     void Print()
12     {
13         std::cout << "show people info:" << std::endl;
14     }
15 
16     void PrintInfo()
17     {
18         std::cout << "name:" << mName << std::endl;
19         std::cout << "age:" << mAge << std::endl;
20     }
21  
22 private:
23  
24     std::string mName;
25     int mAge;
26  
27 };
28  
29 int main()
30 {
31     CPeople* jon = NULL;
32     jon->Print();  // 程序正常运行
33     jon->PrintInfo();  // 程序崩溃,访问非法地址,此时mName和mAge并没有分配空间
34     return 0;
35 }
五、总结

引用网上关于this指针的一个经典回答:
当你进入一个房子后,你可以看见桌子、椅子、地板等,但是房子你是看不到全貌了。对于一个类的实例来说,你可以看到它的成员函数、成员变量,但是实例本身呢?this是一个指针,它时时刻刻指向你这个实例本身。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值