一文详解C++中隐含的this指针

 更新时间:2024年01月29日 10:07:48   作者:仍在探索未知中  
这篇文章主要带大家详细了解一下C++中隐含的this指针,文中通过代码示例和图文介绍的非常详细,对大家的学习或工作有一定的帮助,需要的朋友可以参考下

一、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(2022, 5, 11);
	d2.Init(2022, 5, 12);

	d1.print();
	d2.print();
	return 0;
}

可以看到分别打印出了两个日期,它是怎么知道改打印哪个的?

在这里插入图片描述

我们来分析一下,先看一下汇编代码,看一看

在这里插入图片描述

  • 我们看到上面的代码是调用的同一个函数,那么编译器是如何知道那两个日期的?
  • 其实C++里有一个隐形this指针,在微软的官方文档也有说明
  • 在使用函数的时候其实里面传了一个地址,然后有一个隐的this指针来接收

原形是这样:

在这里插入图片描述

在这里插入图片描述

  • 那为什么这里报错了呢?
  • 因为不必要写,这个是隐含的~~,我们可以直接在类里面使用

二、this指针的特性

刚刚上面也给你看了原形,细心的烙铁已经发现了,这个this指针是有一个const修饰的,而且这个const是在*的右边

在这里插入图片描述

这里的const修饰

在这里插入图片描述

  • 到这里就得出的this本身是不被修改的,但是做指向的值是可以被修改
  • 我们可以在类中打印一下this指针的地址,再打印一下d1和d2的地址,我们来看一下:

在这里插入图片描述

  • 得到结果就是this指针指向一个指向当前对象的指针
  • 我们还可以下面这样,是不会报错的,但是不能向上面直接在形参就写上

在这里插入图片描述

特点:

1、形参和实参的位置,我们不能显示写

2、函数内部可以使用

最后总结一下:

  • this指针的类型:类型 *const,即成员函数中,不能给this指针赋值
  • 只能在“成员函数”的内部使用this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给this形参。所以对象中不存储this指针。
  • this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传递,不需要用户传递

【面试题】

this指针存在哪里?

a、堆 b、栈 c、静态区 d、常量区 e、对象里面

  • 首先排除e,因为我们知道,在算类里的对象的时候是没有算this指针的大小,所以排除e
  • C++的const变量不是在常量区,可以看到这两个地址是挨着的

在这里插入图片描述

那么什么在常量区呢?是const修饰的值在常量区,这个指针变量在栈区,指向了这个常量区的字符串的首字符,所以d也就排除

在这里插入图片描述

  • c就更不可能了,static和全局的才在静态区
  • a也可以排除,因为malloc的才在堆,这里不是malloc,所以排除
  • 最后就是在栈上,因为是一个形参(有些编译比如vs可能会用寄存器存储)。不同的编译器放在不同的位置,可能是栈,也可能是寄存器,(VC++编译器是放在ECX中,其它编译器有可能不同,也就是成员函数的其它参数正常都是存放在栈中。而this指针参数则是存放在寄存器中。)
  • 打开汇编我们也可以看到这里的lea就是load effective address【加载有效地址】,是存在ecx的值加载到 [d1]

在这里插入图片描述

this指针可以为空吗?

下面我们来看两道题来解决这个问题的答案~

第一道:下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行

class A
{
public:
	void Print()
	{
		cout << "Print()" << endl;
	}
private:
	int _a;
};
int main()
{
	A* p = nullptr;
	p->Print();
	return 0;
}

已经完美运行了,因为我这里没有访问类里的对象,所以可以正常运行

在这里插入图片描述

第二道:下面程序编译运行结果是? A、编译报错 B、运行崩溃 C、正常运行

class A
{
public:
	void PrintA()
	{
		cout << _a << endl;

	}
private:
	int _a;
};

int main()
{
	A* p = nullptr;
	p->PrintA();
	return 0;
}

这里引发了空指针,因为我需要打印这里的_a,就要找到那块空间

在这里插入图片描述

也就可以写成这样,this指针是空指针,解引用就会报错

在这里插入图片描述

所以this指针是可以为空的,只要在成员函数内部不访问其内容,程序可以正常执行的。

以上就是一文详解C++中隐含的this指针的详细内容,更多关于C++ this指针的资料请关注脚本之家其它相关文章!

相关文章

  • 基于openCV实现人脸检测

    基于openCV实现人脸检测

    这篇文章主要为大家详细介绍了基于openCV实现人脸检测的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • 数据结构之AVL树详解

    数据结构之AVL树详解

    这篇文章主要介绍了数据结构之AVL树详解,本文非常细致的讲解了AVL树的基础知识、AVL树的旋转操作、AVL数的插入和删除操作等,需要的朋友可以参考下
    2014-08-08
  • C++实现LeetCode(66.加一运算)

    C++实现LeetCode(66.加一运算)

    这篇文章主要介绍了C++实现LeetCode(66.加一运算),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C语言八大排序之堆排序

    C语言八大排序之堆排序

    堆排序是利用堆这种数据结构而设计的一种排序算法,堆排序是一种选择排序,它的最坏,最好,平均时间复杂度均为O(nlogn),它也是不稳定排序
    2022-02-02
  • Opengl ES之FBO帧缓冲对象使用详解

    Opengl ES之FBO帧缓冲对象使用详解

    这篇文章主要为大家介绍了Opengl ES之FBO帧缓冲对象使用详解,<BR>有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-09-09
  • C++超详细讲解强制类型转换

    C++超详细讲解强制类型转换

    在C++语言中新增了四个关键字static_cast、const_cast、reinterpret_cast和dynamic_cast。这四个关键字都是用于强制类型转换的
    2022-05-05
  • OpenCV实现图像背景虚化效果原理详解

    OpenCV实现图像背景虚化效果原理详解

    相信用过相机的同学都知道虚化特效,这是一种使焦点聚集在拍摄主题上,让背景变得朦胧的效果。本文将详细介绍一些这一效果的实现原理以及代码,需要的可以参考一下
    2022-03-03
  • C语言数据结构之队列的定义与实现

    C语言数据结构之队列的定义与实现

    队列是一种特殊的线性表,特殊之处在于它只允许在表的前端(head)进行删除操作,而在表的后端(tail)进行插入操作。本文将详细讲讲C语言中队列的定义与实现,感兴趣的可以了解一下
    2022-07-07
  • c++实现md5加密的代码

    c++实现md5加密的代码

    这篇文章主要介绍了c++实现md5加密的实例代码,代码简单易懂,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-06-06
  • Matlab实现绘制有气泡感的网络图

    Matlab实现绘制有气泡感的网络图

    这篇文章主要介绍了如何利用Matlab实现绘制有气泡感的网络图,文中的示例代码讲解详细,对我们学习Matlab有一定的帮助,需要的可以参考一下
    2023-02-02

最新评论