要弄清楚“为什么基类的析构函数要声明为虚函数”,首先要清楚基类、子类构造函数和析构函数的执行顺序。构造函数的执行顺序为先执行基类的构造函数、再执行子类的构造函数;析构函数的执行顺序为先执行子类的析构函数,再执行基类的析构函数。
如果在基类的析构函数没有声明为虚函数,则在下面这种情况,会有问题
#include <iostream>
#include <windows.h>
using namespace std;
class CBase
{
public:
CBase()
{
std::cout << "CBase" << endl;
}
~CBase()
{
std::cout << "~CBase" << endl;
}
private:
int m_iPara;
};
class CChild : public CBase
{
public:
CChild()
{
std::cout << "CChild" << endl;
}
~CChild()
{
std::cout << "~CChild" << endl;
}
private:
int m_iPara;
};
int main()
{
CBase* p = new(std::nothrow) CChild();
if (nullptr != p)
{
delete p;
p = nullptr;
}
system("pause");
}
运行结果为
可以看到,子类的析构函数没有执行。一个类的析构函数往往和构造函数相对应,构造函数负责一些初始化,析构函数进行反初始化操作,而子类的析构函数没有执行的话,子类析构函数的反初始化操作(往往是一些内存等资源的释放等)就没有执行,从而造成程序异常。
基类的析构函数声明为虚函数后
#include <iostream>
#include <windows.h>
using namespace std;
class CBase
{
public:
CBase()
{
std::cout << "CBase" << endl;
}
virtual ~CBase()
{
std::cout << "~CBase" << endl;
}
private:
int m_iPara;
};
class CChild : public CBase
{
public:
CChild()
{
std::cout << "CChild" << endl;
}
~CChild()
{
std::cout << "~CChild" << endl;
}
private:
int m_iPara;
};
int main()
{
CBase* p = new(std::nothrow) CChild();
if (nullptr != p)
{
delete p;
p = nullptr;
}
system("pause");
}
运行结果为
可以看到子类及基类的析构函数都被正常执行了
标签:函数,子类,析构,基类,执行,构造函数 来源: https://blog.csdn.net/huashuolin001/article/details/110505594
本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享; 2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关; 3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关; 4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除; 5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。