C++分析如何用虚析构与纯虚析构处理内存泄漏
目录
- 一、问题引入
- 二、利用虚析构解决
- 三、利用纯虚析构解决
- 四、总结
一、问题引入
使用多态时,如果有一些子类的成员开辟在堆区,那么在父类执行完毕释放后,没有办法去释放子类的内存,这样会导致内存的泄漏。如下代码段。
如果没有堆区的数据,可以不写虚析构或纯虚析构。
#include <iostream> #include <string> using namespace std; class Animal{ public: Animal(){ cout<<"Animal-构造"<<endl; } ~Animal(){ cout<<"Animal-析构"<<endl; } virtual void Run()=0; //纯虚函数无需实现,只需声明 }; class Cat:public Animal{ public: string *s_name; Cat(string name){ s_name = new string(name);//在堆区创建内存 cout<<"Cat-构造"<<endl; } void Run() { cout<<*s_name<<"->"<<"Cat-Run"<<endl; } ~Cat(){ cout<<"Cat-析构"<<endl; if(s_name!=NULL){ delete s_name; s_name=NULL; } } }; int main() { Animal *a; a = new Cat("Tom"); a->Run(); delete a; //父类指针析构的时候,不会调用子类析构函数 return 0; }
运行结果:
结果可以看到都有父类和子类的构造,虽然在主函数中delete 父类,但最终只有父类的析构函数,此时子类在堆区创建的s_name并没有得到释放,导致内存泄漏。
以上的问题我们引入虚析构和纯虚析构来解决——父类指针释放子类对象时不干净的问题
二、利用虚析构解决
虚析构只需要在析构函数前加关键字 virrtual 即可,再观察结果,可以看到父类和子类的都执行了析构函数,而子类中在堆区创建的数据也被释放干净,这是最终的结果!
virtual ~Animal(){ cout<<"Animal析构"<<endl; }
三、利用纯虚析构解决
纯虚析构格式如下,和纯虚函数有点类似,但需要有具体的声明和具体的实现。纯虚析构需要在类外实现.
class Animal{ public: Animal(){ cout<<"Animal-构造"<<endl; } //虚析构 /*virtual ~Animal(){ cout<<"Animal析构"<<endl; }*/ //纯虚析构 virtual ~Animal()=0; virtual void Run()=0; //纯虚函数无需实现,只需声明 }; //需要有声明,也需要有实现 Animal::~Animal(){ cout<<"纯虚析构"<<endl; }
结果如下,和虚析构有相同的作用
四、总结
虚析构和纯虚析构
相同点: 都可以解决父类指针释放子类对象,都需要有具体的实现
不同点: 纯虚析构属于抽象类,无法实例化对象
到此这篇关于C++分析如何用虚析构与纯虚析构处理内存泄漏的文章就介绍到这了,更多相关C++内存泄漏内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!
相关推荐
-
C++ 内存泄漏调试方式
C++和其他高级语言不同,需要自行管理内存,项目大\调用多,容易内存泄漏.内存申请释放使用malloc/free和new delete两种方式. 当内存报错弹出一个陌生的地址,我一脸懵逼,不清楚是哪个地方出的错,特别是程序退出的时候报错更不好定位.可以使用第三方工具来定位解决,也可以自己想办法解决. 我在做神经网络训练的时候碰到一个的BUG,发送完训练图片之后直接提示使用了非法内存. HEAP[Test.exe]: HEAP: Free Heap block 0000015E950E4A60 m
-
C++内存泄漏及检测工具详解
首先我们需要知道程序有没有内存泄露,然后定位到底是哪行代码出现内存泄露了,这样才能将其修复. 最简单的方法当然是借助于专业的检测工具,比较有名如BoundsCheck,功能非常强大,相信做C++开发的人都离不开它.此外就是不使用任何工具,而是自己来实现对内存泄露的监控,分如下两种情况: 一. 在 MFC 中检测内存泄漏 假如是用MFC的程序的话,很简单.默认的就有内存泄露检测的功能. 我们用VS2005生成了一个MFC的对话框的程序,发现他可以自动的检测内存泄露.不用我们做任何特殊的操作. 仔细
-
C++虚析构函数的使用分析
在C++中,不能声明虚构造函数,但可以声明虚析构函数.多态性是指不同的对象对同一消息有不同的行为特性.虚函数作为运行时多态性的基础,主要是针对对象的,而构造函数是在对象产生之前运行的,因此虚构造函数是没有意义的.析构函数的功能是在该类对象消亡之前进行一些必要的清理工作,析构函数最好都是virtual的.首先解释一下虚构函数和指针之间是如何交互的,以及虚析构函数的具体含义.例如以下代码,其中SomeClass是含有非virtual析构函数的一个类:SomeClass *p= new SomeCla
-
C++之内存泄漏排查详解
目录 一 .经验排查 二 .使用Visual Leak Detector for Visual C++ 2.1 Visual Leak Detector for Visual C++简介 2.2 Visual Leak Detector源码获取编译 2.2.1 源码获取,相关git地址 2.2.2 发布版本获取 2.2.3 进行编译 2.2.4 自带gtest工程测试 2.3 如何测试自己的项目呢 2.3.1 配置工程 2.3.2 编写简单的测试用例 2.3.3 检测结果如图 三.总结 一 .经
-
详谈C++的内存泄漏问题
1)有多少new就有多少delete.而且配对的new与delete要尽量在一个函数中.如果子函数中需要返回的数据是通过new来创建的,我的处理方式一般是在需要调用这个函数的位置将对应的数据规模创建好,并且通过指针或是引用传递到子函数中. 2)本人比较粗心大意,又一次在编码的过程中 竟然将delete语句放在了return语句的后面,导致程序运行的过程中会因内存不足二崩溃.检查了不下20遍才检查出来,真是哭的心情都有了.希望大家以我为戒,一定把return语句放在函数的最后面,估计也就我一个人犯
-
C++内存泄漏的检测与实现详细流程
目录 内存泄漏 带来的问题 难点 hook实现泄漏判断与追踪(malloc和free重载) 宏定义实现hook 内存泄漏 malloc/new 调用在堆上分配的内存却没有相应的free/delete: 带来的问题 会逐渐吃掉虚拟内存 难点 如何判断是否内存泄漏 (最毛糙)可以使用+1和-1,即当调用分配内存变量+1,释放内存时候,变量-1,进程退出输出的count=0,进程内存没有泄漏,否则有泄漏: 线上版本,做一个配置文件,设置是否存在内存泄漏的一个标志位,平时肉眼观察不出的内存泄漏,在长时间
-
解析C++中虚析构函数的作用
我们知道,用C++开发的时候,用来做基类的类的析构函数一般都是虚函数.可是,为什么要这样做呢?下面用一个小例子来说明: 有下面的两个类: 复制代码 代码如下: class ClxBase{public: ClxBase() {}; virtual ~ClxBase() {}; virtual void DoSomething() { cout << "Do something in class ClxBase!" << endl; };}
-
C++的虚析构详解及实例代码
C++的虚析构 最近准备复习一遍所有的知识点,先从基础开始做起,用几分钟写个继承和析构吧. 父类为A,子类为B,代码如下: class A { public: A() { cout << "构造A"<< endl; } ~A() { cout << "析构A" << endl; } } class B:public A { public: B() { cout << "构造B"<&
-
c++ 排查内存泄漏的妙招
前言 对于c++而言,如何查找内存泄漏是程序员亘古不变的话题:解决之道可谓花样繁多.因为最近要用到QT写程序,摆在我面前的第一个重要问题是内存防泄漏.如果能找到一个简单而行之有效的方法,对后续开发大有裨益.久思终得诀窍,本文就详细介绍我对此问题的应对之策.(文末符完整代码) 如何判断内存有泄漏 内存分配和释放对应的操作是new.delete.如何判断内存是否释放干净?其实判断起来非常简单:一个独立的模块整个生存周期内new的个数和delete的个数相等.用伪代码标示如下: int newCoun
-
C++分析如何用虚析构与纯虚析构处理内存泄漏
目录 一.问题引入 二.利用虚析构解决 三.利用纯虚析构解决 四.总结 一.问题引入 使用多态时,如果有一些子类的成员开辟在堆区,那么在父类执行完毕释放后,没有办法去释放子类的内存,这样会导致内存的泄漏.如下代码段. 如果没有堆区的数据,可以不写虚析构或纯虚析构. #include <iostream> #include <string> using namespace std; class Animal{ public: Animal(){ cout<<"A
-
C++ 虚函数和纯虚函数的区别分析
首先:强调一个概念 定义一个函数为虚函数,不代表函数为不被实现的函数. 定义他为虚函数是为了允许用基类的指针来调用子类的这个函数. 定义一个函数为纯虚函数,才代表函数没有被实现. 定义纯虚函数是为了实现一个接口,起到一个规范的作用,规范继承这个类的程序员必须实现这个函数. 简介 假设我们有下面的类层次: class A { public: virtual void foo() { cout<<"A::foo() is called"<<endl; } }; cl
-
C++中虚函数与纯虚函数的用法
本文较为深入的分析了C++中虚函数与纯虚函数的用法,对于学习和掌握面向对象程序设计来说是至关重要的.具体内容如下: 首先,面向对象程序设计(object-oriented programming)的核心思想是数据抽象.继承.动态绑定.通过数据抽象,可以使类的接口与实现分离,使用继承,可以更容易地定义与其他类相似但不完全相同的新类,使用动态绑定,可以在一定程度上忽略相似类的区别,而以统一的方式使用它们的对象. 虚函数的作用是实现多态性(Polymorphism),多态性是将接口与实现进行分离,采用
-
C++之普通成员函数、虚函数以及纯虚函数的区别与用法要点
普通成员函数是静态编译的,没有运行时多态,只会根据指针或引用的"字面值"类对象,调用自己的普通函数:虚函数为了重载和多态的需要,在基类中定义的,即便定义为空:纯虚函数是在基类中声明的虚函数,它可以再基类中有定义,且派生类必须定义自己的实现方法. 假设我们有三个类Person.Teacher.Student它们之间的关系如下: 类的关系图 普通成员函数 [Demo1] 根据这个类图,我们有下面的代码实现 #ifndef __OBJEDT_H__ #define __OBJEDT_H__
-
c++ 虚函数与纯虚函数的区别(深入分析)
在面向对象的C++语言中,虚函数(virtual function)是一个非常重要的概念.因为它充分体现 了面向对象思想中的继承和多态性这两大特性,在C++语言里应用极广.比如在微软的MFC类库中,你会发现很多函数都有virtual关键字,也就是说, 它们都是虚函数.难怪有人甚至称虚函数是C++语言的精髓. 那么,什么是虚函数呢,我们先来看看微软的解释: 虚函数是指一个类中你希望重载的成员函数,当你用一个基类指针或引用指向一个继承类对象的时候,你调用一个虚函数,实际调用的是继承类的版本. 这个定
-
c++中虚函数和纯虚函数的作用与区别
虚函数为了重载和多态的需要,在基类中是有定义的,即便定义是空,所以子类中可以重写也可以不写基类中的此函数! 纯虚函数在基类中是没有定义的,必须在子类中加以实现,很像java中的接口函数! 虚函数 引入原因:为了方便使用多态特性,我们常常需要在基类中定义虚函数. class Cman { public: virtual void Eat(){--}; void Move(); private: }; class CChild : public CMan { public: virtual void
-
C++ 虚函数与纯虚函数的使用与区别
目录 什么是虚函数: 虚函数的注意事项: 纯虚函数 纯虚函数的注意事项: 虚函数与纯虚函数区别 什么是虚函数: 虚函数 是在基类中使用关键字 virtual 声明的函数,在C++ 语言中虚函数可以继承,当一个成员函数被声明为虚函数之后,其派生类中的同名函数都自动生成为虚函数, 虚函数主要体验C++的多态方面,(多态是参数个数和类型相同而实现功能不同的函数) 为了更好的里面虚函数请看下面的demo #include <iostream> #include <string> using
-
虚函数与纯虚函数(C++与Java虚函数的区别)的深入分析
c++虚函数1.定义:在某基类中声明为 virtual 并在一个或多个派生类中被重新定 义的成员函数 [1]2.语法:virtual 函数返回类型 函数名(参数表) { 函数体 }3.用途:实现多态性,通过指向派生类的基类指针,访问派生类中同名覆盖成员函数,也就是允许子类override父类同名方法.虚函数的作用是实现动态联编,也就是在程序的运行阶段动态地选择合适的成员函数,在定义了虚函数后,可以在基类的派生类中对虚函数重新定义,在派生类中重新定义的函数应与虚函数具有相同的形参个数和形参类型(也
-
C++ 虚函数与纯虚函数代码详解
目录 什么是虚函数: 虚函数的注意事项: 存虚函数 总结 什么是虚函数: 虚函数 是在基类中使用关键字 virtual 声明的函数,在C++ 语言中虚函数可以继承,当一个成员函数被声明为虚函数之后,其派生类中的同名函数都自动生成为虚函数, 虚函数主要体验C++的多态方面,(多态是参数个数和类型相同而实现功能不同的函数) 为了更好的里面虚函数请看下面的demo #include <iostream> #include <string> using namespace std; cla
-
C++虚函数表和虚析构介绍
目录 1.虚函数表 2.虚析构 1.虚函数表 虚函数表是C++实现多态的基础,多态是面向对象的三大特性之一,多态有利于提高代码的可读性,便于后期代码的扩展和维护.我们都知道多态的实现是基于虚函数表,那么虚函数表是什么时候创建的呢?虚函数表是怎么实现多态的功能的呢? 首先应该明确多态也称为动态多态,他是在程序运行时候确定函数地址的,也就是程序在运行时,如果类成员函数加了virtual关键字,就会建立一个虚函数指针(vfptr)指针指向一个虚函数表,这个虚函数表就保存了虚函数的地址,子类继承父类也自
随机推荐
- Nginx根据不同浏览器语言配置页面跳转的方法
- 详解使用JavaCV/OpenCV抓取并存储摄像头图像
- Java利用HttpClient模拟POST表单操作应用及注意事项
- Objective-C实现身份证验证的方法示例
- 浅谈JS之iframe中的窗口
- php的一些小问题
- Android 中倒计时验证两种常用方式实例详解
- php header函数的常用http头设置
- Javascript isArray 数组类型检测函数
- JavaScript动态改变HTML页面元素例如添加或删除
- JS 加入收藏夹的代码(主流浏览器通用)
- MySQL优化之分区表
- SQL服务器面临的危险和补救.读[十种方法]后感.
- 详解C语言中的#define宏定义命令用法
- jquery tabs的实现代码
- java 值Document解析xml详细介绍
- MFC创建模态对话框和非模态对话框的方法
- Spring Cloud中各组件超时总结
- 详解PHP文件的自动加载(autoloading)
- Spring中property-placeholder的使用与解析详解