关于C++ delete 来释放new分配的内存

简介: 一般在C语言中我们使用malloc和free进行内存分配和释放,但是在C++中增加了一个新的 new和delete 操作来进行,按照C++的说法delete是释放内存但是指针得到保留,防止内存 泄露,并且NEW和DELETE要成对出现。
一般在C语言中我们使用malloc和free进行内存分配和释放,但是在C++中增加了一个新的
new和delete 操作来进行,按照C++的说法delete是释放内存但是指针得到保留,防止内存
泄露,并且NEW和DELETE要成对出现。我们知道指针本生也是一个保存在内存中某个位置的变量,
如果释放了内存我们是否可以考虑为其中的值得到了删除,而指针自身可以再次指向其他的值?
而还有一点我们需要明白使用NEW分配的内存是HEAP而变量的赋值是栈,


在OS中我们大概可以理解如下(32位系统为例),我这里的共享是指线程是否共享:


4G
     kernel --内核内存
3G   text --代码文本 共享
     data --初始化的全局变量和静态变量 共享
     dss  --未始化的全局变量和静态变量 共享
     栈   --级动态变量数组等 不共享
     堆   --malloc 共享
0    共享库 --库文件 mmap 映射 共享


可以看到栈和堆不是一个区域,并且栈始终是自我释放的遵循后入先出原则


我们接下来用如下的小程序带上GDB进行调试


  8 #include
  9 using namespace std;
 10 
 11 
 12 int main(void)
 13 {
 14     int *p;  //一个不初始化的指针,不能使用*p=一个INT数字,只能赋予一个指针变量p=(init *)0x130000表示p指针指向130000位置
 15     short *b  = new short;
 16     short *c = new short;
 17 
 18     *b=0X128;
 19 
 20     cout<<"noinit int address is :"<< p <<" "<<sizeof(*p)<<endl;
 21     cout<<"short b address is :"<< b <<" "<<sizeof(*b)<<"\n";
 22     cout<<"short c address is :"<< c <<" "<<sizeof(*c)<<endl;
 23     delete b; //成对出现
 24     delete c;
 25     //DELETE不释放指针只是释放内存,那么指针指向的内存可以用于其他用途
 26     cout<<"init int address is :"<< b <<" "<<sizeof(*b)<<"\n";
 27     cout<<"noinit dou address is :"<< c <<" "<<sizeof(*c)<<endl;
 28 
 29     *b=0X256; //直接赋予值,赋值成功
 31     delete b; //不成对出现
 32     *b=0X512; //便于观察而已
 33     delete b; //便于观察,避免指针释放  
  }
~ 如果我们在
 23     delete b;      
 后去GDB *b的只为0那么说明delete起到了效果             
 gdb ./a.out 
(gdb) b 23
Breakpoint 1 at 0x400a79: file pointer2.cpp, line 23.
(gdb) r


Breakpoint 1, main () at pointer2.cpp:23
23              delete b;
(gdb) p b
$1 = (short *) 0x602010
(gdb) x/2xh 0x602010 
0x602010:       0x0128  0x0000
可以看到当前为0X0128数据在这个存储2个字节里面我是也就是0x602010 0x602011
(gdb) n
24          delete c;
(gdb) x/2xh 0x602010
0x602010:       0x0000  0x0000
这里跑完了delete b;  可以看到数据没有了在内存中,继续
继续向下


29              *b=0X256;
(gdb) n
30          delete b;
(gdb) p b
$3 = (short *) 0x602010
这里跑完了 *b=0X256;但是指针位置没有变化,也就是DELETE后指针得到了保留
继续
(gdb) p *b
$4 = 598
(gdb) x/2xh 0x602010
0x602010:       0x0256  0x0000


可以看到数据没有问题。继续看看是否能DELETE
(gdb) n
(gdb) x/2xh 0x602010
0x602010:       0x2020  0x0060
可以看到delete并没有删除数据而已弄了一些垃圾数据进来。
其实这个程序直接跑会报错
*** Error in `./a.out': double free or corruption (fasttop): 0x0000000000f1f010 ***
Aborted (core dumped)
用GDB只是为了找到原因


所以我们必须new和delete 成对使用,否者结果是不确定的,对空指针delete是安全的
delete后指针是得到了保留的没有问题,如果是动态数组将不能使用sizeof来确定他的长度
指针我感觉是使用的栈,同时在函数结束时自我释放。


附带关于GDB的内存查看
转自:http://www.cnblogs.com/super119/archive/2011/03/26/1996125.html


格式: x /nfu


说明
x 是 examine 的缩写


n表示要显示的内存单元的个数


f表示显示方式, 可取如下值
x 按十六进制格式显示变量。
d 按十进制格式显示变量。
u 按十进制格式显示无符号整型。
o 按八进制格式显示变量。
t 按二进制格式显示变量。
a 按十六进制格式显示变量。
i 指令地址格式
c 按字符格式显示变量。
f 按浮点数格式显示变量。


u表示一个地址单元的长度
b表示单字节,
h表示双字节,
w表示四字节,
g表示八字节




Format letters are o(octal), x(hex), d(decimal), u(unsigned decimal),
t(binary), f(float), a(address), i(instruction), c(char) and s(string).
Size letters are b(byte), h(halfword), w(word), g(giant, 8 bytes)


举例
x/3xh buf
表示从内存地址buf读取内容,
h表示以双字节为一个单位,
3表示三个单位,
x表示按十六进制显示


</sizeof(*c)<<endl;
</sizeof(*b)<<"\n";
</sizeof(*c)<<endl;
</sizeof(*b)<<"\n";
</sizeof(*p)<<endl;
相关实践学习
阿里云图数据库GDB入门与应用
图数据库(Graph Database,简称GDB)是一种支持Property Graph图模型、用于处理高度连接数据查询与存储的实时、可靠的在线数据库服务。它支持Apache TinkerPop Gremlin查询语言,可以帮您快速构建基于高度连接的数据集的应用程序。GDB非常适合社交网络、欺诈检测、推荐引擎、实时图谱、网络/IT运营这类高度互连数据集的场景。 GDB由阿里云自主研发,具备如下优势: 标准图查询语言:支持属性图,高度兼容Gremlin图查询语言。 高度优化的自研引擎:高度优化的自研图计算层和存储层,云盘多副本保障数据超高可靠,支持ACID事务。 服务高可用:支持高可用实例,节点故障迅速转移,保障业务连续性。 易运维:提供备份恢复、自动升级、监控告警、故障切换等丰富的运维功能,大幅降低运维成本。 产品主页:https://www.aliyun.com/product/gdb
相关文章
|
2月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
124 26
|
8月前
|
存储 程序员 编译器
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
317 0
|
10月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
865 13
|
11月前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
515 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
10月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
543 4
|
11月前
|
存储 Java
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
这篇文章详细地介绍了Java对象的创建过程、内存布局、对象头的MarkWord、对象的定位方式以及对象的分配策略,并深入探讨了happens-before原则以确保多线程环境下的正确同步。
190 0
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
|
11月前
|
程序员 C语言 C++
C++入门5——C/C++动态内存管理(new与delete)
C++入门5——C/C++动态内存管理(new与delete)
205 1
|
2月前
|
存储
阿里云轻量应用服务器收费标准价格表:200Mbps带宽、CPU内存及存储配置详解
阿里云香港轻量应用服务器,200Mbps带宽,免备案,支持多IP及国际线路,月租25元起,年付享8.5折优惠,适用于网站、应用等多种场景。
726 0
|
2月前
|
存储 缓存 NoSQL
内存管理基础:数据结构的存储方式
数据结构在内存中的存储方式主要包括连续存储、链式存储、索引存储和散列存储。连续存储如数组,数据元素按顺序连续存放,访问速度快但扩展性差;链式存储如链表,通过指针连接分散的节点,便于插入删除但访问效率低;索引存储通过索引表提高查找效率,常用于数据库系统;散列存储如哈希表,通过哈希函数实现快速存取,但需处理冲突。不同场景下应根据访问模式、数据规模和操作频率选择合适的存储结构,甚至结合多种方式以达到最优性能。掌握这些存储机制是构建高效程序和理解高级数据结构的基础。
209 0

热门文章

最新文章