当类型转换void指针指向对象时,析构函数不会被调用

3 c++

样品

void func(void* data)
{
 CResource* resource = (CResource*)data;
 delete resource; // ~CResource never called.
 resource = NULL;
}
Run Code Online (Sandbox Code Playgroud)

请帮我弄明白这一点.

Sum*_*uma 11

总结了可能无法调用CResource析构函数的原因,从其他答案中提取:

不完整的类型

一个可能的原因是您只声明了CResource类型,未定义:

class CResource;

void func(void* data)
{
 CResource* resource = (CResource*)data;
 delete resource; // ~CResource never called.
 resource = NULL;
}
Run Code Online (Sandbox Code Playgroud)

这是一个未定义的行为(删除不完整的类型).在这样的情况下,编译器应该发出关于未调用析构函数的警告(Visual C++肯定会发出它).如果是这种情况,请确保在要销毁它的位置定义了类型(包括所需的标题).

空指针

如果data为NULL,则delete不执行任何操作,也不会调用任何析构函数.

类型不匹配

如果CResource析构函数是虚拟的,并且存储在数据指向的内存中的对象实际上是不同的类型,则会得到未定义的行为.通常会调用不同的析构函数(如果对象具有另一个虚拟析构函数),在其他情况下程序可能会崩溃(如果对象没有虚拟析构函数).


Pau*_*ius 6

导致析构函数不被调用的唯一原因是数据指针是0(或NULL).这就是删除的工作原理 - 它检查指针是否为0,如果不是 - 它调用必要的析构函数并释放内存.

正如评论中指出的那样.还有另一个原因就是它不会被调用.如果数据指向某个其他类(而不是CResource)对象,并且这两个类都有虚拟析构函数,那么将调用该另一个类的析构函数.