对于虚析构函数的理解
2017-08-06 16:09
246 查看
首先,对于虚析构函数,那就得说下构造函数和析构函数了。
构造函数:进行初始化成员变量的函数。
析构函数:在对象生命周期结束的时候,完成资源的回收和清理。
如果我们在设计一个类的时候,没有显示声明定义构造函数,析构函数,则编译器会自动生成。
虚析构函数:只有当一个类被定义为基类的时候,才会把析构函数写成虚析构函数。
如果我们不需要使用基类对派生类的对象操作时,我们也不必去定义虚析构函数,这样会增加系统的内存开销,当类里面有虚析构函数时,系统会为当前类分配一个虚函数表,里面存放虚函数指针,这样就会增加类的存储空间。关于虚函数表的一些知识,可以访问:http://blog.csdn.net/haoel/article/details/1948051/ 。
对于虚析构,就是在析构函数前加virtual关键字,那么到底有什么作用呢?
防止内存泄露,定义一个基类的指针p,在delete p时,如果基类的析构函数是虚函数,这时只会看p所赋值的对象,如果p赋值的对象是派生类的对象,就会调用派生类的析构函数(毫无疑问,在这之前也会先调用基类的构造函数,在调用派生类的构造函数,然后调用派生类的析构函数,基类的析构函数,所谓先构造的后释放);如果p赋值的对象是基类的对象,就会调用基类的析构函数,这样就不会造成内存泄露。
如果基类的析构函数不是虚函数,在delete p时,调用析构函数时,只会看指针的数据类型,而不会去看赋值的对象,这样就会造成内存泄露。
下面是具体的代码演示:
仔细对比上面两个代码的运行结果,我们就不难得出上面的结论了。
所以我们编程时候切记在析构函数前加virtual关键字。
最后,文章如果有瑕疵的地方,请大神指出。
构造函数:进行初始化成员变量的函数。
析构函数:在对象生命周期结束的时候,完成资源的回收和清理。
如果我们在设计一个类的时候,没有显示声明定义构造函数,析构函数,则编译器会自动生成。
虚析构函数:只有当一个类被定义为基类的时候,才会把析构函数写成虚析构函数。
如果我们不需要使用基类对派生类的对象操作时,我们也不必去定义虚析构函数,这样会增加系统的内存开销,当类里面有虚析构函数时,系统会为当前类分配一个虚函数表,里面存放虚函数指针,这样就会增加类的存储空间。关于虚函数表的一些知识,可以访问:http://blog.csdn.net/haoel/article/details/1948051/ 。
对于虚析构,就是在析构函数前加virtual关键字,那么到底有什么作用呢?
防止内存泄露,定义一个基类的指针p,在delete p时,如果基类的析构函数是虚函数,这时只会看p所赋值的对象,如果p赋值的对象是派生类的对象,就会调用派生类的析构函数(毫无疑问,在这之前也会先调用基类的构造函数,在调用派生类的构造函数,然后调用派生类的析构函数,基类的析构函数,所谓先构造的后释放);如果p赋值的对象是基类的对象,就会调用基类的析构函数,这样就不会造成内存泄露。
如果基类的析构函数不是虚函数,在delete p时,调用析构函数时,只会看指针的数据类型,而不会去看赋值的对象,这样就会造成内存泄露。
下面是具体的代码演示:
#include <iostream> using namespace std; class Base { private: int i; public : Base() { cout << "Base count " << endl; } virtual ~Base() { cout << " Base descount" << endl; } }; class Inherit :public Base { private: int num; public: Inherit() { cout << "Inherit count" << endl; } ~Inherit() { cout << "Inherit descout" << endl; } }; int main() { Base *p = new Inherit; delete p; Base *q = new Base; delete q; return 0; }
#include <iostream> using namespace std; class Base { private: int i; public : Base() { cout << "Base count " << endl; } ~Base() { cout << " Base descount" << endl; } }; class Inherit :public Base { private: int num; public: Inherit() { cout << "Inherit count" << endl; } ~Inherit() { cout << "Inherit descout" << endl; } }; int main() { Base *p = new Inherit; delete p; Base *q = new Base; delete q; return 0; }
仔细对比上面两个代码的运行结果,我们就不难得出上面的结论了。
所以我们编程时候切记在析构函数前加virtual关键字。
最后,文章如果有瑕疵的地方,请大神指出。
相关文章推荐
- 对于 PHP cookie 与 session 的理解
- 对于一个工程的结构理解
- 对于RAC心跳以及仲裁机制的理解 推荐
- 对于数据类型转换的理解
- 对于脏读,不可重复读,幻读的一点理解,看懂红字很关键
- 对于STM32别名区的理解 (转载)
- 熊先生做原型之对于原型设计中高保真的理解
- 对于php输出INF的理解
- 对于Javascript闭包的理解
- 对于redis框架的理解(三)
- 对于程序员说的话,项目经理们是这样理解的
- 对于继承的理解
- 个人对于冒泡排序和选择排序的理解
- 对于 “OWASP TOP 10” 的一些理解
- 对于程序员说的话,项目经理们是这样理解的
- 对于oauth2.0的个人理解-客户端篇
- 分享张开旭微博对于分词的理解
- 个人对于堆栈的理解(简单通俗)
- 对于swiz1.0框架的一点理解
- 对于MVC的理解