为什么在虚表中需要一个纯虚函数表项
Why an entry of pure virtual function in virtual table is required
我对虚表的理解是,每当编译器在类中找到虚函数时,它就会为该类创建一个虚表,并且虚函数的所有函数指针都将放在该表中。
但是当涉及到纯虚函数时,我们不会在任何时候调用该函数。那么为什么在虚表中需要纯虚函数项呢?
virtual void myFunction() = 0 ;
该声明是必需的,因为您需要告诉编译器在虚函数表中为从声明该方法的基类开始的特定方法保留一个槽位(这是您在派生类上调用方法时可能希望使用的类型)
只是给你一个想法,让我们做一个例子(这是不考虑到底发生了什么)。假设在Base
中有三个虚方法,其中一个是纯方法
class Base {
virtual void pure() = 0;
virtual void nonpure() { }
virtual void nonpure2() { }
};
所以Base
变量表看起来像
0 [ pure ] -> nothing
1 [ nonpure ] -> address of Base::nonpure
2 [ nonpure2] -> address of Base::nonpure2
现在我们用
推导它class Derive : public Base {
virtual pure() override { }
virtual nonpure2() override { }
};
Derived
vtable将看起来像
0 [ pure ] -> address of Derived::pure
1 [ nonpure ] -> address of Base::nonpure
2 [ nonpure2 ] -> address of Derived::nonpure2
当你尝试做
Base* derived = new Derived();
derived->pure();
方法大致编译为
address = derived->vtable[0];
call address
如果您没有在Base
类中声明纯虚方法,那么在编译时就没有办法知道它在虚表中的索引(0),因为该方法根本不存在。
但是,如果虚表中有一个"洞"(缺少实现),则不能实例化该特定的类类型。
不能实例化抽象类的对象。这实际上使您的问题有点无意义:因为您永远不会实例化抽象类,因此根本不需要该类的虚拟表。(实际上,它可能在施工/破坏期间暂时需要,但这是另一回事。)
当你实际实例化一个对象时,它是某个派生的类的对象,它不再是抽象的。它不再有任何纯虚函数了。您实际实例化的派生类在那时已经覆盖了所有纯虚函数。这就是在虚方法表中需要一个表项的原因——用来存储指向实际重写函数的指针。
在后面的代码中,可以通过指向抽象基类 的指针调用myFunction()
。MyAbstractBaseClass *ptr = some_function();
// Pointer actually points to some non-abstract derived object
ptr->myFunction();
编译器将生成代码,进入与*ptr
对象相关联的虚方法表,提取对应于myFunction()
的指针条目,并通过该指针传递控制权。如上所述,该指针实际上将指向某个派生类的重写函数。
相关文章:
- 创建一个函数以在输入为负数或零时输出字符串.第一次执行用户定义的函数
- 如何仅为一个函数添加延迟
- 有没有什么方法可以使用一个函数中定义的常量变量,也可以由c++中同一程序中的其他函数使用
- 我需要将多个函数组合为一个函数
- 在C++中声明一个函数时,它需要有函数本身的参数吗
- 如何创建一个函数来计算并返回平均值、最大值和最小值
- 一个函数,用于查找字符串1包含字符串2 c++的次数
- 如何将一个类的函数作为另一个类的另一个函数的参数传递
- 编写一个函数以使用 n 百分比的 CPU 使用率
- 将 N-arg 函数包装到另一个函数中
- 如何封装一个函数,以便它只能由同一类中的一个其他函数调用?
- C++(.cpp文件和.h文件)拆分代码并添加一个函数,提取 - 这很容易吗?
- C++从另一个函数退出函数
- 编写一个函数来删除单链表中的节点(尾部除外),仅授予对该节点的访问权限
- 视觉我希望一个函数在另一个函数C++中进行计算
- C ++如何在原始抽象类中创建一个函数,该函数接受派生类的输入
- 在另一个函数 (c++) 中调用变量
- 如何在另一个函数中使用返回值作为参数?
- 如何包装一个函数以适应另一个函数的所需类型
- 创建一个函数的 Python 绑定,返回指向带有 boost 的向量的指针