我知道,对于存在虚函数的基类,一般需要将基类的析构函数定义为虚函数,从而实现资源的合理释放,而且我也知道派生类在重载时,只需要清理自己的对象,不过,有时候还是会有些疑惑感,所以写了一个简单的例子,来消除疑惑。下面是实例内容:
#include <iostream>
class student
{
public:
student() {}
~student() { std::cout << "a student" << std::endl; }
};
class bachelor
{
public:
bachelor() {}
~bachelor() { std::cout << "a bachelor" << std::endl; }
};
class studentHolder
{
public:
studentHolder()
{
}
virtual ~studentHolder() {}
private:
student st;
};
class bachelorHolder : public studentHolder
{
public:
bachelorHolder()
: studentHolder()
{
}
~bachelorHolder() override
{
}
private:
bachelor bcl;
};
下面是调用的地方的代码:
#define _CRTDBG_MAP_ALLOC
#include <stdlib.h>
#include <crtdbg.h>
int main()
{
_CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF);
studentHolder* holder = new bachelorHolder();
delete holder;
return 0;
}
这种情况下,会执行正确的清理操作:
这种情况下,如果将bachelorHolder中的析构函数删除,即:
class bachelorHolder : public studentHolder
{
public:
bachelorHolder()
: studentHolder()
{
}
private:
bachelor bcl;
};
这种情况下,对象依然被很好的清理,因为C++会默认为类创建一个析构函数,而且如果基类为虚函数,派生类创建的为重写基类的虚函数。结果依然是:
如果在上述的情况下,将studentHolder析构函数的virtual移除,即:
class studentHolder
{
public:
studentHolder()
{
}
~studentHolder() {}
private:
student st;
};
结果将变成:
不过,注意一点,这里,只是说派生类的析构函数没有被调用,资源(分配的堆内存)还是会被很好的释放。这里,可以查看,如果我们将main函数中的delete holder;注释掉:
int main()
{
_CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF);
studentHolder* holder = new bachelorHolder();
// delete holder;
return 0;
}
在debug模式下,进行调试,会在output窗口,得到如下结果:
而就算将main函数,改为如下:
int main()
{
_CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF);
studentHolder* holder = new bachelorHolder();
delete reinterpret_cast<void*>(holder);
return 0;
}
在output窗口,都不会出现内存泄漏的显示。大致内容就是上述显示,在有疑虑的时候,可以参考一下。
原文链接: https://www.cnblogs.com/albizzia/p/8979078.html
欢迎关注
微信关注下方公众号,第一时间获取干货硬货;公众号内回复【pdf】免费获取数百本计算机经典书籍
原创文章受到原创版权保护。转载请注明出处:https://www.ccppcoding.com/archives/273140
非原创文章文中已经注明原地址,如有侵权,联系删除
关注公众号【高性能架构探索】,第一时间获取最新文章
转载文章受原作者版权保护。转载请注明原作者出处!