浅谈C++中的那些内存泄露

浅谈C++中的那些内存泄露

大家好,又见面了,我是全栈君。

     尽管学过C语言。可是C++里面的一些基础还是不太懂,还须要再掌握。

老范也開始要讲C++设计模式了,必须快点看了。不然就要白花窝滴钱了。

     对于内存泄露,我的个人理解就是程序在执行过程中,自己开辟了空间,用完这块空间后却没有释放。

今晚上我就犯了这种低级错误,导致程序没有执行出来。还是先看代码:

#include <iostream>

/* run this program using the console pauser or add your own getch, system("pause") or input loop */
using namespace std;
class person{
	public:
		person(){
			cout<<"基类构造函数运行中.....\n";
		}
	       ~person(){
			cout<<"基类析构函数运行中.....\n";
		}
}; 
class DS:public person{
	public:
		DS(){
			cout<<"派生类构造函数运行中.....\n";
		}
		~DS(){
			cout<<"派生类析构函数运行中.....\n";
		}
};
int main(int argc, char** argv) {
	
	DS p;
	return 0;
}

这段代码是没有问题的,程序開始从main函数执行,实例化派生类DS一个对象p。无论怎样。派生类DS总先调用基类person的构造函数,然后派生类DS在调用自己的构造函数,接着是自己的析构函数,最后是基类person的构造函数,执行结果例如以下图所看到的:

浅谈C++中的那些内存泄露

事实上嘛,这个什么时候调基类构造(析构)。什么时候调派生类构造(析构)。我个人的理解能够用简单的图来表示了。基类构造和析构就像一个大框架包括着派生类的构造和析构:

浅谈C++中的那些内存泄露

 

1. 对于上述的程序我在main函数中继续改动,假设new了。却没delete。(用匹配和兼容任意,结果还是一样的)

int main(int argc, char** argv) {
	

	DS *p=new DS();
        // person *p= new DS();
	return 0;
}

这时候就出现故障了,假设在C++中。就会出现以下的结果:

浅谈C++中的那些内存泄露

原因是new了,却不没delete,造成内存泄露。在程序执行过程中析构函数是不会调用的,直到整个程序结束。系统才会自己主动释放内存。

2. 对于上述的程序我在main函数中继续改动,这次加上delete p,如今匹配的情况下操作。也就是派生类的指针指向派生对象:

int main(int argc, char** argv) {
	

	DS *p=new DS();
	 delete  p;
	return 0;
}

执行结果大家也应该猜的出来,4个全有

浅谈C++中的那些内存泄露

 

3. 对于上述的程序我在main函数中继续改动,这次利用兼容,即就是基类指针指向派生类的对象:

<strong>int main(int argc, char** argv) {
	

	 person *p=new DS();
	 delete  p;
	return 0;
}</strong>

 

可是这次又出现故障了,执行结果中没有派生类的析构函数,这都是兼容惹的祸,由于基类指针仅仅能指向派生类继承自己的那一部分,对于派生类DS那一部分,基类指针映射不到。所以就不去调用它,演示结果例如以下:

浅谈C++中的那些内存泄露

要解决这问题实际不难。这个时候就要。基类指针就基类指针,管他三七二十一的。我这个时候仅仅要在基类person的析构函数中加一个virtual(虚特性)。尽管在main函数中,还是第三中情况。我也能够轻轻松松的输出以下的结果:

浅谈C++中的那些内存泄露

事实上这个里面涉及到了。继承(兼容规则),多态的知识,在C++涉及模式中,95%都用到了多态性,无疑这是C++的重点,一定要好好学习这一块。

对于继承和多态的相关知识下一个博客在说吧,那个就要大写特写了,,哈哈,那是一个非常有意思的举例——媳妇炒菜,工厂模式,慢慢的就进入C++的核心中去了。。

接下来。关于C++中,指针delete删除值得问题。这一块有必要说一下:

先把代码发上去在说吧

#include <iostream>

/* run this program using the console pauser or add your own getch, system("pause") or input loop */
using namespace std;
int main(int argc, char** argv) {
	int *p;
	p=new int;
	*p=3;
     cout<<"输出值*p= "<<*p<<endl;
     cout<<"输出地址p= "<<p<<endl;
     
     delete p;   //delete p 之后 
	 
     cout<<"delete p 之后输出值*p= "<<*p<<endl;
     cout<<"delete p 之后输出地址p= "<<p<<endl;
     
     	  
     
	return 0;
}

我定义了一个整型指针p。new了一块int型的整型单元,并给*p赋值3,之后呢,输出*p和p的地址。之后我在delete p。事实上我们每次delete实际上是删除的是p指向那块空间的值。并没有删除它的地址,因此才会出现以下的结果演示:

浅谈C++中的那些内存泄露

好了,我的能力有限就仅仅能写到这里了,以后遇到问题在完好,不正确的地方也要改动。,睡觉了,,。瞌睡死了,。,,

        

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/115715.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • java中保留两位小数的输出

    java中保留两位小数的输出例如平方和与倒数和,最后输出要求保留两位小数,下面这个程序的注意点已用红色字体标记importjava.text.DecimalFormat;importjava.util.Scanner;publicclassMain{publicstaticvoidmain(String[]args){//TODOAuto-generatedmet

  • 使用R中merge()函数合并数据[通俗易懂]

    使用R中merge()函数合并数据[通俗易懂]使用R中merge()函数合并数据在R中可以使用merge()函数去合并数据框,其强大之处在于在两个不同的数据框中标识共同的列或行。如何使用merge()获取数据集中交叉部分merge()最简单的形式为获取两个不同数据框中交叉部分。举例,获取cold.states和large.states完全匹配的数据。代码如下:&gt;merge(cold.states,large….

  • Excel 增加删除线

    Excel 增加删除线Excel增加删除线操作如下:2.选中删除线即增加删除线,去掉删除线即删除删除线,颜色可以定义成红色之类。

  • 博弈论基础mooc答案_博弈论考试题及答案

    博弈论基础mooc答案_博弈论考试题及答案1、“博弈的本意是什么?A、摔跤B、下棋C、赌博D、游戏参考答案:B2、古时“弈”字,就是指A、跳棋B、象棋C、五子棋D、围棋参考答案:D3、按照博弈方是否达成有约束力的协议,可以分为()A、理性博弈和非理性博弈B、完全信息博弈和不完全信息博弈C、动态博弈和静态博弈D、合作博弈与非合作博弈参考答案:D4、囚徒困境的例子属于()的典型A、非合作博弈B、合作博弈C、理性博弈D、动态博弈参考答案:A5、“石头剪刀布游戏,属于()。A、贯序博弈B、动态博弈…

    2022年10月15日
  • Mybatis面试题(总结最全面的面试题!!!)

    Mybatis面试题(总结最全面的面试题!!!)什么是数据持久化?数据持久化是将内存中的数据模型转换为存储模型,以及将存储模型转换为内存中的数据模型的统称。例如,文件的存储、数据的读取等都是数据持久化操作。数据模型可以是任何数据结构或对象的模型、XML、二进制流等。当我们编写应用程序操作数据库,对表数据进行增删改查的操作的时候就是数据持久化的操作。Mybatis框架简介MyBatis框架是一个开源的数据持久层框架。它的内部封装了…

  • mysql的端口是多少_如何查看db2数据库的端口

    mysql的端口是多少_如何查看db2数据库的端口查看mysql端口号(mysql端口号是多少)2020-05-0722:11:45共10个回答如何查看mysql的端口号1使用命令showglobalvariableslike’port’;查看端口号2修改端口,编辑/etc/my.cnf文件,早期版本有可能是my.conf文件名,增加端口参数,并且设定端口,注意该端口未被使用,保存退出.总结:注意修改的端口不要被占用,而且要有规划,不要轻意的总…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号