C++的内存管理详细解释

目录
  • 一、C/C++内存分布
  • 二、C语言中动态内存管理方式:
    • 1、malloc/calloc/realloc区别:
  • 三、C++中动态内存管理:new/delete
  • 四、实现原理
  • 五、面试常问问题
    • 1、malloc/free和new/delete的区别
    • 2、内存泄漏
      • 内存泄漏分类(了解)
  • 总结

一、C/C++内存分布

  • 栈又叫堆栈,非静态局部变量/函数参数/返回值等等,栈是向下增长的。
  • 内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享共享内存,做进程间通信。
  • 堆用于程序运行时动态内存分配,堆是可以向上增长的。
  • 数据段–存储全局数据和静态数据。
  • 代码段–可执行的代码/只读常量。

二、C语言中动态内存管理方式:

1、malloc/calloc/realloc区别:

共同点:

  • 都是C语言中用来进行动态内存申请的库函数,申请的空间都在堆上,用完之后必须使用free来进行释放
  • 返回值类型都是void*在接受返回的地址时必须要进行强转
  • 如果申请空间成功:返回的是空间的首地址,如果失败返回的是NULL

不同点:

  • malloc:返回值类型void*

在接受返回的空间地址时必须要进行强转

成功:空间首地址 失败:NULL

参数:申请的空间所占总的字节数

申请的空间在堆上,使用完成后必须要使用free来进行释放

  • calloc:返回值一致

参数列表:参数一表示元素的个数;参数二表示单个类型的字节数

功能:与malloc基本相同,但是calloc会对其申请的空间进行初始化

  • realloc(void* p, size_t size):将p所指向空间的大小调整到size字节

p指向的是NULL:该函数的类似malloc

假设:p所指向的空间总共占old个字节

size <= old:将p所指向的空间缩小到size个字节---->直接返回值p所指向空间的首地址

size > old:将p所指向的空间扩增到size个字节

大一点:返回原空间的首地址

大的多的多:申请新空间;将旧空间中元素拷贝到新空间;释放旧空间;返回新空间的首地址

三、C++中动态内存管理:new/delete

C++为什么要搞一套动态内存管理?

首先:C语言中的动态内存管理方式在C++中仍然可以使用

原因:

1、C语言中的方式比较麻烦—需要用户手动算字节数,需要对返回结果强转,需要判空,需要包含头文件

2、malloc、free:不会调用构造函数/析构函数;new、delete:在进行空间申请和释放时是会调用构造函数和析构函数的

// C++中动态内存管理方式:new/delete----申请单个类型的空间
// new[]/delete[]-----申请释放一段连续的空间
//注意:1、new/delete不是函数,是C++中的关键字||操作符
//		2、new的空间必须要有delete释放  new[]必须使用delete[]释放
class Test
{
public:
	Test()
		:_t(10)
	{
		cout << "Test():" << this << endl;
	}
	~Test()
	{
		cout << "~Test():" << this << endl;
	}
	int _t;
};
//new/delete和new[]/delete[]使用说明
void Test1()
{
	int* p1 = new int;
	int* p2 = new int(100);
	int* p3 = new int[10];
	int* p4 = new int[10]{ 1,2,3,4,5,6,7,8,9,0 };
	delete p1;
	delete p2;
	delete[] p3;
	delete[] p4;
}
void Test2()
{
	//malloc并不是创建了一个Test类型的对象,只是在堆上申请了一块与Test类型大小相同的一块空间
	//因为:malloc不调用构造函数
	Test* p1 = (Test*)malloc(sizeof(Test));//malloc不会调用构造函数
	if (nullptr == p1)
		return;
	//真正创建了一个对象,该对象的空间在堆上
	Test* p2 = new Test;//new在申请空间期间会调用构造函数
	free(p1);//:在释放空间期间,不会调用析构函数
	delete p2;//:在释放空间期间,会调用对象的析构函数
}
/*
在C++中,如果想要在堆上申请空间:
1、采用C语言中的malloc、calloc、realloc,但是并不能申请对象的空间
2、采用new/new[]---可以调用构造函数,注意:如果使用new[]申请连续的空间是,该类必须提供无参或全缺省的构造函数
3、malloc/free,new/delete,new[]/delete[]必须成对使用,否则会内存泄漏或者代码崩溃
*/

注意:申请和释放单个元素的空间,使用new和delete操作符,申请和释放连续的空间,使用new[]和delete[]

四、实现原理

new的原理

1、调用operator new函数申请空间

2、在申请的空间上执行构造函数,完成对象的构造

delete的原理

1、在空间上执行析构函数,完成对象中资源的清理工作

2、 调用operator delete函数释放对象的空间

new T[N]的原理

1、调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申

2、在申请的空间上执行N次构造函数

delete[]的原理

1、 在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理

2、调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释放空间

五、面试常问问题

1、malloc/free和new/delete的区别

共同点是

都是从堆上申请空间,并且需要用户手动释放。

不同的地方是:

  • malloc和free是函数,new和delete是操作符
  • malloc申请的空间不会初始化,new可以初始化
  • malloc申请空间时,需要手动计算空间大小并传递,new只需在其后跟上空间的类型即可
  • malloc的返回值为void*, 在使用时必须强转,new不需要,因为new后跟的是空间的类型
  • malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new需要捕获异常
  • 申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数与析构函数,而new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间中资源的清理

2、内存泄漏

什么是内存泄漏:内存泄漏指因为疏忽或错误造成程序未能释放已经不再使用的内存的情况。内存泄漏并不是指内存在物理上的消失,而是应用程序分配某段内存后,因为设计错误,失去了对该段内存的控制,因而造成了内存的浪费。

内存泄漏的危害:长期运行的程序出现内存泄漏,影响很大,如操作系统、后台服务等等,出现内存泄漏会导致响应越来越慢,最终卡死

内存泄漏分类(了解)

C/C++程序中一般我们关心两种方面的内存泄漏:

1、堆内存泄漏(Heap leak)

堆内存指的是程序执行中依据须要分配通过malloc / calloc / realloc / new等从堆中分配的一块内存,用完后必须通过调用相应的 free或者delete 删掉。假设程序的设计错误导致这部分内存没有被释放,那么以后这部分空间将无法再被使用,就会产生Heap Leak。

2、系统资源泄漏

指程序使用系统分配的资源,比方套接字、文件描述符、管道等没有使用对应的函数释放掉,导致系统资源的浪费,严重可导致系统效能减少,系统执行不稳定。

如何避免内存泄漏

1、工程前期良好的设计规范,养成良好的编码规范,申请的内存空间记着匹配的去释放。ps:这个理想状态。但是如果碰上异常时,就算注意释放了,还是可能会出问题。需要下一条智能指针来管理才有保证。

2、 采用RAII思想或者智能指针来管理资源。

3、有些公司内部规范使用内部实现的私有内存管理库。这套库自带内存泄漏检测的功能选项。

4、出问题了使用内存泄漏工具检测。ps:不过很多工具都不够靠谱,或者收费昂贵。

解决方案:

1、事前预防型。如智能指针等。

2、事后查错型。如泄漏检测工具。

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • 关于C/C++内存管理示例详解

    1.内存分配方式 在C++中,内存分成五个区,分别是堆.栈.自由存储区.静态存储区和常量存储区. 1) 栈 执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放.栈内存分配运算内置处理器指令集中,效率很高,但分配的内存容量有限. 2) 堆 由new分配的内存块,释放由程序员控制.如果程序员没有释放,那么就在程序结束的时候,被操作系统回收. 3) 自由存储区 由malloc等分配的内存块,用free结束自己的生命. 4) 静态存储区 全局变量和静态变量被分配到

  • C++动态内存管理详解

    目录 1.C/C++程序地址空间 2.C语言动态内存管理 (1)malloc (2)calloc (3)realloc (4)free 3.C++动态内存管理 (1)C++为什么要设计一套自己专属的动态内存管理方式? (2)new/delete定义 1)new/delete操作内置类型 2)new/delete操作自定义类型 (3)new/delete的实现原理 4.malloc/free和new/delete的区别 共同点: 不同点: 5.内存泄漏 总结 1.C/C++程序地址空间 计算机物理

  • 养成良好的C++编程习惯之内存管理的应用详解

    开篇导读    虽然本系列文章定位为科普读物,但本座相信它们不但适合新手们学习借鉴,同时也能引发老鸟们的反思与共鸣.欢迎大家提出宝贵的意见和反馈 ^_^ 在开篇讲述本章主要内容之前,本座首先用小小篇幅论述一下一种良好的工作习惯 -- 积累.提炼与求精.在工作和学习的过程中,不断把学到的知识通过有效的方式积累起来,形成自己的知识库,随着知识量的扩大,就会得到从量变到质变的提升.另外还要不断地对知识进行提炼,随着自己知识面的扩大以及水平的提升,你肯定会发现原有知识库存在着一些片面.局限.笨拙甚至错误

  • 详解C/C++内存管理

    C/C++赋予程序员管理内存的自由,是C/C++语言特色,虽然这引入了复杂度和危险性,但另一方面,它也增加了控制力和灵活性,是C/C++独特之处,亦是强大之处. C/C++内存分布 让我们先来看看下面这段代码: int globalVar = 1; static int staticGlobalVar = 1; void Test() { static int staticVar = 1; int localVar = 1; int num1[10] = { 1, 2, 3, 4 }; char

  • C++的内存管理详细解释

    目录 一.C/C++内存分布 二.C语言中动态内存管理方式: 1.malloc/calloc/realloc区别: 三.C++中动态内存管理:new/delete 四.实现原理 五.面试常问问题 1.malloc/free和new/delete的区别 2.内存泄漏 内存泄漏分类(了解) 总结 一.C/C++内存分布 栈又叫堆栈,非静态局部变量/函数参数/返回值等等,栈是向下增长的. 内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库.用户可使用系统接口创建共享共享内存,做进程间通信.

  • C++内存管理详细解析

    目录 一.C++内存管理 1. new/delete表达式 2.new/delete重载 3.类内自定义allocator(per-class allocator) 二.多线程内存分配器 1.malloc/free 2.brk和mmap 三.补充知识 1.内存泄漏 2.malloc/free和new/delete的比较 3.RAII规则 一.C++内存管理 C++中有四种内存分配.释放方式: 最高级的是std::allocator,对应的释放方式是std::deallocate,可以自由设计来搭

  • javascript内存管理详细解析

    介绍 低层次的语言,如C,具有低级别的内存管理命令,如:malloc()和free(),需要开发者手工释放内存.然而像javascript这样的高级语言情况则不同,对象(objects, strings 等)创建的时候分配内存,当他们不在使用的时候内存会被自动回收,这个自动回收的过程被称为垃圾回收.因为垃圾回收的存在,让javascript等高级语言开发者产生了一个错误的认识,以为可以不用关心内存管理. 内存生命周期 不管什么样的编程语言,内存的生命周期基本上是一致的. 1.分配你需要的内存 2

  • linux 内存管理机制详细解析

    物理内存和虚拟内存我们知道,直接从物理内存读写数据要比从硬盘读写数据要快的多,因此,我们希望所有数据的读取和写入都在内存完成,而内存是有限的,这样就引出了物理内存与虚拟内存的概念. 物理内存就是系统硬件提供的内存大小,是真正的内存,相对于物理内存,在linux下还有一个虚拟内存的概念,虚拟内存就是为了满足物理内存的不足而提出的策略,它是利用磁盘空间虚拟出的一块逻辑内存,用作虚拟内存的磁盘空间被称为交换空间(Swap Space). 作为物理内存的扩展,linux会在物理内存不足时,使用交换分区的

  • Linux内存管理和寻址详细介绍

    目录 1.概念 内存管理模式 地址类型划分 说明: 2.页式管理 x86架构32位cpu x86架构 64位cpu 3.地址划分 4. 调试 结语 1.概念 内存管理模式 段式:内存分为了多段,每段都是连续的内存,不同的段对应不用的用途.每个段的大小都不是统一的,会导致内存碎片和内存交换效率低的问题. 页式:内存划分为多个内存页进行管理,如在 Linux 系统中,每一页的大小为 4KB.由于分了页后,就不会产生细小的内存碎片.但是仍然也存在内存碎片问题. 段页式:段式和页式结合. 地址类型划分

  • C语言 超详细梳理总结动态内存管理

    目录 一.为什么存在动态内存分配 二.动态内存函数的介绍 1.malloc和free 2.calloc 3.realloc 三.常见的动态内存错误 1.对NULL指针的解引用操作 2.对动态开辟空间的越界访问 3.对非动态开辟的空间使用free释放 4.使用free释放一块动态开辟空间的一部分 5.对同一块开辟的空间多次释放 6.动态内存开辟忘记释放(内存泄漏) 四.几个经典的笔试题 一.为什么存在动态内存分配 我们已经掌握的内存开辟方式有: int a = 10://在栈空间开辟4个字节的连续

  • 超详细分析C语言动态内存管理问题

    目录 一.为什么存在动态内存的分配 二.动态内存函数的介绍 2.1 malloc和free 2.2 calloc 2.3 realloc 三.常见的动态内存错误 3.1 对NULL指针的解引用操作 3.2 对动态开辟空间的越界访问 3.3 对非动态开辟内存使用free释放 3.4 对同一块动态内存多次释放 3.5 动态开辟内存忘记释放(内存泄漏) 四.几个经典的笔试题 五.C/C++程序的内存开辟 六.柔性数组 6.1 柔性数组的特点 6.2 柔性数组的使用 6.3 柔性数组的优势 上期结束了[

  • C语言与C++内存管理超详细分析

    目录 一.内存 1.1 内存四区 1.2 使用代码证实内存四区的底层结构 二.malloc 和 free 2.1 malloc 和 free 的使用 2.2 内存泄漏与安全使用实例与讲解 三.new 和 delete 3.1 new 和 delete 使用 3.2 delete 与 delete[] 的区别 一.内存 在计算机中,每个应用程序之间的内存是相互独立的,通常情况下应用程序 A 并不能访问应用程序 B,当然一些特殊技巧可以访问,但此文并不详细进行说明.例如在计算机中,一个视频播放程序与

  • C语言详细分析讲解内存管理malloc realloc free calloc函数的使用

    目录 C语言内存管理 一.动态空间申请 二.动态空间的扩容 三.释放内存 C语言内存管理 malloc && realloc && free && calloc c语言中为了进行动态内存管理,<stdlib.h>中提供了几个函数帮助进行内存管理. 我们知道,C语言中是没有C++中的容器或者说是python中list,set这些高级的数据结构的,我们一旦申请了一段内存空间以后这一段空间就归你了,比如我们举个例子,我们申请一个数组 int nums[

  • C++详细讲解内存管理工具primitives

    目录 primitives new 和 delete placement new 重载 operator new per-class allocator New Handler =default,=delete primitives 分配 释放 属于 是否可重载 malloc() free() C 不可 new delete C++表达式 不可 ::operator new() ::operator delete() C++函数 可 allocator::allocate() allocator

随机推荐