为什么C++不按运行时类型销毁对象?
Why doesn't C++ destruct the object by run-time type?
,据我所知,当基类具有虚拟函数时,C 可以获取对象动态类型的准确信息。
class Base
{
public:
Base() {}
~Base() { std::cout << "Base Destructed" << std::endl; }
virtual void f() {}
};
class Derived : public Base
{
public:
Derived() {}
~Derived() { std::cout << "Derived Destructed" << std::endl; }
};
void PrintTypeName(Base *p)
{
std::cout << typeid(*p).name() << std::endl;
}
int main()
{
Base *p = new Derived();
PrintTypeName(p);
delete p;
}
上面的代码可以打印正确的对象类型,但是为什么它不能调用正确的驱动器。
我在G 和Windows编译器上测试了它,它们给出了相同的结果。我知道如果我制造基本驱动器virtual
,它可以正确地破坏。
,但我想知道为什么不通过typeid
调用破坏者。
如果未将方法标记为virtual
,则不会存储在虚拟表中。在这种情况下,驱动器不会存储在虚拟表中。因此无法称呼。
使用当前代码, vtables 将看起来像这样:
vtable for Derived:
.quad 0
.quad typeinfo for Derived
.quad Base::f()
vtable for Base:
.quad 0
.quad typeinfo for Base
.quad Base::f()
请参阅演示此处
如果destructor在Base
中标记为virtual
,则 vtables 将看起来像这样:
vtable for Derived:
.quad 0
.quad typeinfo for Derived
.quad Derived::~Derived() [complete object destructor]
.quad Derived::~Derived() [deleting destructor]
.quad Base::f()
vtable for Base:
.quad 0
.quad typeinfo for Base
.quad Base::~Base() [complete object destructor]
.quad Base::~Base() [deleting destructor]
.quad Base::f()
请参阅演示此处
如果您打算从中派生类,请务必记住要制作基类驱动器virtual
。
在基类中将其虚拟标记将告诉编译器,它也要在删除基本对象时也要破坏派生对象。
class Base
{
public:
Base() {}
virtual ~Base() { std::cout << "Base Destructed" << std::endl; }
virtual void f() {}
};
但是要注意的是,您不会将虚拟的功能放在函数面前,而您不打算覆盖。因为这附带了维护虚拟表的额外成本。
https://www.crashhandler.com/2019/03/make-destructors-destroy-virtally.html这是我对此的博客文章。
相关文章:
- 在 C++ 中将一个模板类型的对象类型转换为另一个模板类型
- 了解 Python 中的对象类型
- 具有纯虚函数和指针数组对象类型的父类的指针数组
- 调用的对象类型 'double' 不是 report() 函数的函数或函数指针
- 基准、值、值类型、对象和对象类型(C++)
- JNI 如何将 Java 对象数组传递给相同对象类型的 C++ 数组
- 在构造过程中更改的对象类型
- 如何构建程序以避免查询对象类型?
- 如何使用Dynamic_cast获得对象类型
- 什么更有效率?在重载函数中或通过在基类函数中检查对象类型来实现
- static_cast实际上不是对象类型的类型是未定义的行为吗?
- 为什么 std::variant 不能容纳数组对象类型,而联合可以?
- 为什么类型转换对象不会更改其地址?有关对象类型的信息存储在哪里?
- 类返回对象类型,但如何返回和检查 null
- 是否可以在辅助功能中概括对象类型
- 表达式必须在C 中具有指针对象类型
- 下标需要数组或指针类型表达式必须具有指针对象类型
- 当将对象传递给具有参数作为引用类型的函数以及当其类对象类型时,会得到不同的输出
- 如何在运行时指定对象类型
- 我可以获取在重载的新运算符中使用新运算符的对象类型吗?