C++语法|虚函数与多态详细讲解(二)|深入理解虚函数

本文主要内容来自《CPU眼里的C/C++》

系列汇总讲解,请移步:
C++语法|虚函数与多态详细讲解系列(包含多重继承内容)

从汇编级别剖析虚函数

首先我们定义了一个类A,其中包含一个普通函数 func() 和虚函数 vfunc()
在这里插入图片描述

我们可以看到,这俩函数在汇编上面其实没有任何区别!

就跟阿布老师说的一样,首先要破除在语法层面上的神秘!

函数调用的静态绑定和动态绑定

接下来,我们就应该看它的调用情况了:
在这里插入图片描述
普通函数和虚函数的调用明显不一样,

普通成员函数func在调用时,直接call A::func(),这也就说明在编译阶段,我们就已经知道要调用哪个函数了,这就是所谓的静态绑定

然而调用虚函数的时候,我们的 call 在根据 rdx 寄存器中的值来确定函数位置,rdx的值从何而来呢?我们看红色方框中最上面的三条指令:

mov		rax, QWORD PTR [rbp-8]
mov		rax, QWORD PTR [rax]
mov		rax, QWORD PTR [rax]

前两条命令就是在获取对象实例 a 的虚函数表地址(vptr),第三条指令就是在获取 vfunc 的内存地址(该函数位于虚函数表的第一项,第零项是我们的RTTI)。然后我们将其存在 rdx 寄存器中,马上 call rdx 完成函数的调用,其实这也就是所谓的动态绑定

也就是说在C++ 语法层面,如果编译器发现被调函数是虚函数,他就会通过 vptr 到虚函数表找到该函数的地址,完成调用。

综上,我们需要注意的是,在一个类的内存结构中,不仅存在成员变量、成员函数的地址、还存在一个虚函数表指针 vptr,并且它位于类对象内存结构中的第零个位置

小总结:

继承结构中的虚函数

我们定义一个派生类B,我们看以下汇编代码:
在这里插入图片描述
我们可以明显的看到,如果A有派生类B,B在调用自己的构造函数(这里的构造函数是编译器为我们自动生成的),并且 vptr 被初始化成B的虚函数表地址,从而保证A、B的虚函数相互独立。

并且需要格外注意,尽量不要在构造函数中调用虚函数,因为如果在指向派生类B的构造函数时,还会先调用基类A的构造函数,所以 vptr 会一会被初始化为A的虚函数表,一会儿被初始化成B的虚函数表,这时候虚函数的动态绑定特性失效,会跟语法规则产生一定的冲突,不过听说高手仍然有处理办法。

总结:
(1)虚函数的调用,需要借助类对象的隐藏变量 vptr 来完成,该变量会在构造函数中被初始化成虚函数表的内存地址。
(2)如果该类没有虚函数,编译器在分配类的对象资源时还是按照成员函数和成员变量来的;一旦该类有虚函数,编译器就会在构造函数中完成 vptr 的初始化,并且把起始的4字节(32位)字节留给 vptr 。

  • 12
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
下面是一个使用C++虚函数实现多态的代码示例: ```cpp #include <iostream> using namespace std; class A { public: virtual void foo() { cout << "A::foo()" << endl; } virtual ~A() { cout << "~A()" << endl; } }; class B : public A { public: void foo() { cout << "B::foo()" << endl; } ~B() { cout << "~B()" << endl; } }; int main() { A* p = new B(); p->foo(); delete p; return 0; } ``` 在这个示例中,类A和类B都有一个名为foo()的虚函数。当我们通过基类指针p调用foo()函数时,实际上会根据指针所指向的对象的类型来确定调用哪个类的foo()函数。这就是多态的实现原理。 输出结果为: ``` B::foo() ~B() ~A() ``` 可以看到,通过虚函数实现的多态,调用的是派生类B的foo()函数,而不是基类A的foo()函数。同时,析构函数也是虚函数,确保在删除指针p时,会先调用派生类B的析构函数,再调用基类A的析构函数。这是因为在C++中,如果基类的析构函数不是虚函数,那么通过基类指针删除派生类对象时,只会调用基类的析构函数,而不会调用派生类的析构函数。因此,为了确保正确的析构顺序,析构函数应该声明为虚函数。 请注意,虚函数只能在类的成员函数中声明,构造函数不能声明为虚函数。这是因为在创建对象时,需要先调用构造函数来初始化对象,而虚函数的机制是在运行时根据对象的类型来确定调用哪个函数,而构造函数在对象创建时就已经确定了。因此,构造函数不能声明为虚函数。 #### 引用[.reference_title] - *1* [虚函数实现多态原理](https://blog.csdn.net/qq_24309981/article/details/89102183)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [C++虚函数多态实现](https://blog.csdn.net/qq_27576655/article/details/124535530)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值