理解 C++ 的 this 指针

1. 什么是 this 指针

**C++**中的 this 指针是一个由系统维护的指针,其指针本身的地址存储在某个内存位置,对程序员不可见(程序员不能取得其地址),其仅在当前定义的实体的函数体内可见。

当任何外部函数调用实体的函数时,系统会将该实体的地址存入 this 指针,进入函数后就可以通过该 this 指针访问实体的成员(数据和函数)。

2. 例释 this 指针(以windows 64位平台为例)

为了简单起见,我们定义一个结构体并定义一个函数,使用默认构造函数:

struct DataEntity

{

__int64 m = 10;

__int64 n = 20;

__int64 Add()

{

return m + n;

}

};

调用代码:

DataEntity data;

__int64 total = data.Add();

实现分析(汇编代码): 下面通过其生成的汇编代码来分析其实现过程。

(1) 调用构造函数

DataEntity data;

00007FF689FF5073 lea rcx,[data] ;将实体的地址存入寄存器 rcx (调用构造函数传参用)

00007FF689FF5078 call DataEntity::DataEntity (07FF689E76F55h) ;调用构造函数

下面进入构造函数:

DataEntity::DataEntity(void):

00007FF7946A13F0 mov qword ptr [this],rcx ;将实体地址存入this指针

;在每一个生成函数中,都会在进行正式操作前插入这样一段代码,先将实体地址存入this指针,

;以便程序员后续利用 this指针操作该对象的成员。

00007FF7946A13F5 mov rax,qword ptr [this] ;取得this存储的对象地址

00007FF7946A13FA mov qword ptr [rax],0Ah ; 将 10 送入 m ,因为此时寄存器 rax

;存放的是实体的首地址,其首地址就是 m 的地址。

00007FF7946A1401 mov rax,qword ptr [this] ;再取this存储的对象地址存入寄存器 rax

00007FF7946A1406 mov qword ptr [rax+8],14h ;地址+8正好是n的内存位置,将20送入

;n 的地址指向的内存

00007FF7946A140E mov rax,qword ptr [this] ; 将 this 指针送入寄存器 rax 以备使用

00007FF7946A1413 ret ; 操作完成,函数返回

注意:该 this 指针在函数调用期间保持不变。

(2) 调用自定义函数

__int64 total = data.Add();

00007FF79481507D lea rcx,[data] ;将实体的地址存入寄存器 rcx (调用构造函数传参用)

00007FF794815082 call DataEntity::Add (07FF794696F64h) ;调用函数

00007FF794815087 mov qword ptr [total],rax ;从rax取出计算结果送入total 内存。

进行被调函数内部流程:

__int64 Add()

{

00007FF7946A12F0 mov qword ptr [this],rcx ; 取出实体地址存入 this

return m + n;

00007FF7946A12F5 mov rax,qword ptr [this] ;得取this

00007FF7946A12FA mov rax,qword ptr [rax] ; m 值送入寄存器 rax

00007FF7946A12FD mov rcx,qword ptr [this] ;得取this

00007FF7946A1302 add rax,qword ptr [rcx+8] ;rcx+8=n的地址,将n加到rax(m)

}

00007FF7946A1306 ret ;函数返回,寄存器rax的值即为计算结果,上述标红的部分即从rax取

;出结果,在 Windows X64 加构下,约定使用 rax 传递返回值。

相关推荐
weixin_4723394622 分钟前
高效处理大体积Excel文件的Java技术方案解析
java·开发语言·excel
枯萎穿心攻击1 小时前
响应式编程入门教程第二节:构建 ObservableProperty<T> — 封装 ReactiveProperty 的高级用法
开发语言·unity·c#·游戏引擎
Eiceblue2 小时前
【免费.NET方案】CSV到PDF与DataTable的快速转换
开发语言·pdf·c#·.net
tan180°3 小时前
MySQL表的操作(3)
linux·数据库·c++·vscode·后端·mysql
m0_555762903 小时前
Matlab 频谱分析 (Spectral Analysis)
开发语言·matlab
浪裡遊4 小时前
React Hooks全面解析:从基础到高级的实用指南
开发语言·前端·javascript·react.js·node.js·ecmascript·php
彭祥.4 小时前
Jetson边缘计算主板:Ubuntu 环境配置 CUDA 与 cudNN 推理环境 + OpenCV 与 C++ 进行目标分类
c++·opencv·分类
lzb_kkk5 小时前
【C++】C++四种类型转换操作符详解
开发语言·c++·windows·1024程序员节
好开心啊没烦恼5 小时前
Python 数据分析:numpy,说人话,说说数组维度。听故事学知识点怎么这么容易?
开发语言·人工智能·python·数据挖掘·数据分析·numpy
简佐义的博客5 小时前
破解非模式物种GO/KEGG注释难题
开发语言·数据库·后端·oracle·golang