从汇编的角度揭开C++ this指针的神秘面纱(下)

<接上篇>

我们接着来看一段C++的代码:

复制代码
class A
{
public:
    int func(int j)
    {
        return j +_i;
    }
private:
    int _i;    
};

int main()
{
    A a;
    a.func(3);
    return 0;
}

这里定义了一个类A,在main函数中定义了A类的对像a。同时用a调用了成员函数func。我们来看一下main函数的汇编代码:

复制代码
        pushq   %rbp
        movq    %rsp, %rbp
        subq    $16, %rsp         //rsp = rsp - 16
        leaq    -4(%rbp), %rax    //rax = rpb -4
        movl    $3, %esi          //esi = 3
        movq    %rax, %rdi        //rdi = rax
        call    A::func(int)
        movl    $0, %eax
        leave
        ret

在调用A::func之前,赋值了两个寄存器:rdi和rsi. 由上篇文单可知,这两个寄存器用于函数调用时,传递第1和第2个参数。那么这里我们有疑问:A::func(int)明明只有一个参数,为什么调用时,传递了两个参数呢?另一个参数是什么呢?答案就是:在调用成员函数时,this指针会做为一个隐含的参数传递给成员函数,并且是作为第一个参数。那么this指针又代表什么呢,this指针就是指向类对象的一个地址。对应上面汇编代码即是rbp-4。源代码a.func(3)相当于func(this, 3)也就是func(&a, 3);

为了方便理解,我画出了main函数的栈空间,如下所示:

this指针存储的是a对象的地址。即rbp-4. 此时a对象里面只有一个int型的成员变量i, 占用4个字节。

接下来,我们分析一下func函数,汇编如下:

复制代码
A::func(int):
        pushq   %rbp
        movq    %rsp, %rbp
        movq    %rdi, -8(%rbp)      //*(rbp-8) = rdi
        movl    %esi, -12(%rbp)     // *(rbp-12) = esi
        movq    -8(%rbp), %rax      // rax = *(rbp-8) 
        movl    (%rax), %edx        // edx = *rax
        movl    -12(%rbp), %eax     // eax = *(rbp-12)
        addl    %edx, %eax          // eax += edx
        popq    %rbp
        ret

func函数栈空间如下:

func中会先分配栈空间rbp-8, rbp-12来存储传递过来的参数rdi(this)以及esi(3)。 然后根据这些参数来做下一步处理。

总结:

1)this指针的本质即对象的地址。

2)在调用类成员函数时,C++编译器会将其作为函数的第一个参数,传递给成员函数。

相关推荐
皮皮林5511 小时前
SpringBoot 加载外部 Jar,实现功能按需扩展!
java·spring boot
rocksun2 小时前
认识Embabel:一个使用Java构建AI Agent的框架
java·人工智能
Java中文社群3 小时前
AI实战:一键生成数字人视频!
java·人工智能·后端
王中阳Go3 小时前
从超市收银到航空调度:贪心算法如何破解生活中的最优决策谜题?
java·后端·算法
shepherd1113 小时前
谈谈TransmittableThreadLocal实现原理和在日志收集记录系统上下文实战应用
java·后端·开源
维基框架3 小时前
Spring Boot 项目整合Spring Security 进行身份验证
java·架构
日月星辰Ace4 小时前
Java JVM 垃圾回收器(四):现代垃圾回收器 之 Shenandoah GC
java·jvm
天天摸鱼的java工程师5 小时前
商品详情页 QPS 达 10 万,如何设计缓存架构降低数据库压力?
java·后端·面试
天天摸鱼的java工程师5 小时前
设计一个分布式 ID 生成器,要求全局唯一、趋势递增、支持每秒 10 万次生成,如何实现?
java·后端·面试
阿杆5 小时前
一个看似普通的定时任务,如何优雅地毁掉整台服务器
java·后端·代码规范