内存管理(C/C++)

简介: 内存管理(C/C++)

C/C++内存分布


先介绍C/C++中程序内存区域的分划


f5008a21422819f3b6c4027d3960d6db_549a3f9f07bd4e67b3535719308917c9.png


观察下列代码,分析各数据在内存中所处的位置


int i = 1;
static int statici = 1;
void test()
{
  static int staticn = 1;
  int n = 1;
  int arr[10] = { 0 };
  char ch[] = "crush";
  const char* p = "crush";
  int* ptr1 = (int*)malloc(sizeof(int) * 4);
  int* ptr2 = (int*)calloc(4, sizeof(int));
  int* ptr3 = (int*)realloc(ptr2, sizeof(int) * 4);
  free(ptr1);
  free(ptr3);
}


dddad88ddb7710cab3098650f145d292_2538e69416b74b9497763cfd618d24eb.png

A.栈   B.堆   C.数据段(静态区)   D.代码段(常量区)
i在哪里__                    statici在哪里__
staticn在哪里__              n在哪里__
arr[]在哪里__
ch[]在哪里__                 *ch在哪里__
p在哪里__                    *p在哪里__
ptr1在哪里__                 *ptr1在哪里__
sizeof(arr)=__
sizeof(ch)=__               strlen(ch)=__
sizeof(p)=__                strlen(p)=__
sizeof(ptr1)=__

dddad88ddb7710cab3098650f145d292_2538e69416b74b9497763cfd618d24eb.png

7e23e303018e7f481ba890b8bfbdde09_a0ef3ef3e7b64a838b0e53b022c83e46.png



栈:存储非静态局部变量,函数参数,返回值等

堆:程序运行时进行内存分配

数据段:存储全局数据和静态数据

代码段:可执行程序,只读常量


C语言中的动态内存管理方式


malloc/calloc/realloc 和 free

代码如下


void test()
{
  int* ptr1 = (int*)malloc(sizeof(int) * 4);
  int* ptr2 = (int*)calloc(4, sizeof(int));
  int* ptr3 = (int*)realloc(ptr2, sizeof(int) * 4);
  free(ptr1);
  //free(ptr2)  这里需要free(ptr2)吗?
  free(ptr3);
}

由于已经在ptr2的基础上进行了扩容,所以不需要对ptr2进行释放,否则就会发生错误


d3ac642cc7f8feb822d5deb417ca0aaa_eaf0825aec354a1081ee42520362870d.png


C++中动态内存管理


new / delete 操作内置类型


void test()
{
  int* p1 = new int;                   //C++动态申请一个int类型的空间
  int* p2 = (int*)malloc(sizeof(int)); //C动态申请一个int类型的空间
  delete p1;
  free(p2);
  int* p3 = new int[10];                    //C++动态申请10个int类型的空间
  int* p4 = (int*)malloc(sizeof(int) * 10); //C动态申请10个int类型的空间
  delete p3;
  free(p4);
  int* p5 = new int(1);                //C++动态申请一个int类型的空间并初始化
  int* p6 = new int[5]{ 1,2,3,4,5 };   //C++动态申请5个int类型的空间并初始化
  delete p5;
  delete[] p6;
}


new/delete 和 malloc/free 针对内置类型没有任何区别


void test()
{
  //动态申请一个int类型的空间
  int* p1 = new int;
  //动态申请一个int类型的空间并进行初始化
  int* p2 = new int(1);
  //动态申请5个int类型的空间
  int* p3 = new int[5];
  delete p1;
  delete p2;
  delete[] p3;
}


1c57954fac72873568324210159a1c1b_5345f003051e4967a6090ec510d7d4ba.png


C++动态申请和释放空间时,需要配对,若是单个元素,使用new/delete;若是多个元素使用new[]/delete[]


new 和 delete 操作自定义类型


class M
{
public:
  M()
  :_data(1)
  {
  cout << "M()" << endl;
  }
  ~M()
  {
  cout << "~M()" << endl;
  }
private:
  int _data;
};
void test()
{
  //C++动态申请一个M类型的空间
  M* p1 = new M;
  //C动态申请一个M类型的空间
  M* p2 = (M*)malloc(sizeof(M));
  delete p1;
  free(p2);
  //C++动态申请5个M类型的空间
  M* p3 = new M[5];
  //C动态申请5个M类型的空间
  M* p4 = (M*)malloc(sizeof(M) * 5);
  delete[] p3;
  delete p4;
}
int main()
{
  M m;
  test();
  return 0;
}


a9284fa06a1167fe165611b729140b0c_e2dc4bf3784840a08977a4c736666afd.png


在申请自定义类型的空间时,new会调用构造函数,delete也会调用析构函数;相反malloc 和 free 则不会


operator new 和 operator delete 函数


operator new 和 operator delete 函数


new / delete C++是进行动态内存申请和释放的操作符, operator new和 operator delete是系统提供的全局函数,换一种说法便是,new 申请空间是通过调用 operator new来实现的,同理 delete 释放空间也是通过调用operator delete来实现的


本质上,operator new/operator delete的用法与 malloc/free 的功能完全一样,不同点在于,处理动态申请失败的方式不同;malloc申请失败返回NULL,operator new申请失败则会抛异常


以下三种动态申请空间和释放的效果是一样的


void test()
{
  int* p1 = new int;
  delete p1;
  int* p2 = (int*)operator new(sizeof(int));
  operator delete(p2);
  int* p3 = (int*)malloc(sizeof(int));
  free(p3);
}


f2f154dac9d4757ca13d4fd1d1b94d7c_ae794e77477542fe9577d42546676faf.png


new 和 delete 的实现原理


内置类型


申请的是内置类型的空间,new/delete 与 malloc/free 的区别在上面已经结束过,这里便不在赘叙


自定义类型


new的原理


调用operator new函数申请空间

在已经申请的空间上调用构造函数,完成对象的实例化


delete的原理


在已经申请的空间上调用析构函数,完成对象中资源的清理

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


new T[N]的原理


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

在已经申请的空间上调用N次构造函数


delete[]的原理


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

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


定位 new 表达式(placement-new)


定位new表达式是在已经分配的原始内存空间中调用构造函数初始化一个对象


使用格式:

new(place-address)type或者

new(place-address)type(initalizer-list)


使用场景

定位new表达式在实际中一般是配合内存池使用。因为内存池分配出的内存没有初始化,所以如果是自定义类型的对象,需要使用new的定位表达式进行显示调用构造函数并进行初始化


class M
{
public:
  M(int m = 0)
  :_m(1)
  {
  cout << "M()" << endl;
  }
  ~M()
  {
  cout << "~M()" << endl;
  }
private:
  int _m;
};
void test()
{
  //p现在指向的是与M对象大小相同的一段空间
  //还未调用构造函数,所以指向的不算是对象
  M* p = (M*)operator new(sizeof(M));
  //定位new,显示调用构造函数并进行初始化已经申请的空间
  new(p)M;//new(p)M(1);
  //显示调用析构函数
  p->~M();
  operator delete(p);
}


常见面试题


malloc / free 和 new / delete 的区别


malloc/free 和 new/delete的共同点:都是从堆上申请空间,并且使用完之后需要进行释放


malloc 和 free是函数,new和delete是操作符

malloc申请的空间不进行初始化,new可以进行初始化

malloc申请空间时,需要手动计算空间大小并传递,new只需要在其后跟上空间的类型即可,如果多个对象,[]中指定对象个数即可

malloc的返回值是void*,在使用时必须强转,new不需要,因为new后跟的是空间的类型

malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new需要捕获异常

申请自定义类型对象时,malloc/free 只会开辟空间,不会调用构造函数和析构函数,而new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间中资源的清理


内存泄漏


什么是内存泄漏,内存泄漏的危害


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


内存泄漏的危害:长期运行的程序出现泄漏,影响很大。


目录
相关文章
|
4月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
187 26
|
9月前
|
存储 程序员 编译器
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
5月前
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
105 1
|
8月前
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
10月前
|
存储 程序员 编译器
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
485 0
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
211 3
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
614 4
|
存储 C语言 C++
【C++打怪之路Lv6】-- 内存管理
【C++打怪之路Lv6】-- 内存管理
157 0
【C++打怪之路Lv6】-- 内存管理
|
存储 C语言 C++
【C/C++内存管理】——我与C++的不解之缘(六)
【C/C++内存管理】——我与C++的不解之缘(六)
|
程序员 C语言 C++
C++入门5——C/C++动态内存管理(new与delete)
C++入门5——C/C++动态内存管理(new与delete)
240 1
下一篇
oss云网关配置