【C++】| C/C++内存管理

本文涉及的产品
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
简介: 【C++】| C/C++内存管理

前言:

在上期,我们已经对类和对象的全部知识进行了总结和梳理。在类和对象学习完之后,今天我将给大家呈现的是关于——C/C++内存管理的基本知识。


 


1. 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 char2[] = "abcd";
 const char* pChar3 = "abcd";
 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);
}
1. 选择题:
   选项: A.栈  B.堆  C.数据段(静态区)  D.代码段(常量区)
   globalVar在哪里?____   staticGlobalVar在哪里?____
   staticVar在哪里?____   localVar在哪里?____
   num1 在哪里?____
   
   char2在哪里?____       *char2在哪里?___
   pChar3在哪里?____      *pChar3在哪里?____
   ptr1在哪里?____        *ptr1在哪里?____

解析如下:

  • 【globalVar】: globalVar作为全局变量在数据段(静态区)
  • 【staticGlobalVar】:staticGlobalVar作为静态全局变量在静态区
  • 【staticVar】:staticVar作为静态局部变量在静态区
  • 【localVar】: localVar作为局部变量局部变量放在栈区
  • 【num1】: num1为局部变量一样也存放在栈区

以上这个五个我相信大多数小伙伴都可以作对的,接下来我们讲解下面几个:

  • 【char2】:  char2是一个数组,跟num1的区别是(num1是自己确定大小,而char2是通过初始化来确定大小),所以不难得出num1和char2是在一个地方的,即——作为局部变量,放在栈区  
  • 【*char2】:char2是一个数组,把后面常量字符串拷贝过来到数组中,数组在栈上,所以*char2在栈上
  • 【pChar3】: pChar3局部变量在栈区
  • 【* pChar3】: *pChar3得到的是字符串常量字符在代码段
  • 【ptr1】:  ptr1是指针,作为局部变量栈区  
  • 【* ptr1】: *ptr1就是ptr指向的那块空间,因此得到的是动态申请空间的数据在堆区

接下来,我们结合图形,大家可以直观的感受!!

接下来,还有个连环的问题,大家在看看下列的题目,不知道各位是否能够拿下它呢?

sizeof(num1) = ____;  
 sizeof(char2) = ____;   strlen(char2) = ____;
 sizeof(pChar3) = ____;   strlen(pChar3) = ____;
 sizeof(ptr1) = ____;

解析:

  •  sizeof(num1) = __40__; sizeof数组名,即计算数组大小,10个整形数据一共40字节
  •  sizeof(char2) = __5__; 包括\0的空间,因此为5
  •  strlen(char2) = __4__; 遇到\0即截止,不包括\0的长度,因此为4
  •  sizeof(pChar3) = __4__; 因为pChar3为指针,所以跟后面指向的美誉关系,32位下为大小4,64位下8
  •  strlen(pChar3) = __4__; 字符串“abcd”的长度,不包括\0的长度
  •  sizeof(ptr1) = __4__; ptr1是指针,同上

【说明】🤫

  • 1. 栈又叫堆栈--非静态局部变量/函数参数/返回值等等,栈是向下增长的。
  • 2. 内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口 创建共享共享内存,做进程间通信。(Linux课程如果没学到这块,现在只需要了解一下)
  • 3. 堆用于程序运行时动态内存分配,堆是可以上增长的。
  • 4. 数据段--存储全局数据和静态数据。
  • 5. 代码段--可执行的代码/只读常量。

通过上述的问题带大家仔细再次认识了一下程序在内中的分布问题。接下来,我们将探讨关于动态内存管理方式的问题!!!


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

这个知识点,我们在之前就已经具体的讲到过了,在这里给大家简单的在过一遍。这里给大家一段代码,大家先回顾一下之前有关的知识:

void Test ()
{
int* p1 = (int*) malloc(sizeof(int));
free(p1);
// 1.malloc/calloc/realloc的区别是什么?
int* p2 = (int*)calloc(4, sizeof (int));
int* p3 = (int*)realloc(p2, sizeof(int)*10);
// 这里需要free(p2)吗?
free(p3 );
}

(1)C语言跟内存分配方式

  • <1>从静态存储区域分配.
  • 内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在.例如全局变量、static变量.
  • <2>在栈上创建
  • 在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放.栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限.
  • <3>从堆上分配,亦称动态内存分配.
  • 程序在运行的时候用malloc或new申请任意多少的内存,程序员自己负责在何时用free或delete释放内存.动态内存的生存期由用户决定,使用非常灵活,但也很容易出现问题

(2)C语言跟内存申请相关的函数

C语言跟内存申请相关的函数主要有calloc、malloc、free、realloc等.

  • <a>malloc分配的内存是位于堆中的,并且没有初始化内存的内容,因此基本上malloc之后,调用函数memset来初始化这部分的内存空间.
  • <b>calloc则将初始化这部分的内存,设置为0.
  • <c>realloc则对malloc申请的内存进行大小的调整.
  • <d>申请的内存最终需要通过函数free来释放.

切记:

  • 当程序运行过程中malloc了,但是没有free的话,会造成内存泄漏.一部分的内存没有被使用,但是由于没有free,因此系统认为这部分内存还在使用,造成不断的向系统申请内存,使得系统可用内存不断减少.

对于malloc详细的知识可以参考如下地址:

malloc函数

对于calloc详细的知识可以参考如下地址:

calloc函数

对于realloc详细的知识可以参考如下地址:

realloc函数


(3)面试题

接下来给大家解答一个常见的面试题——malloc/calloc/realloc的区别?

区别:

   (1)函数malloc不能初始化所分配的内存空间,而函数calloc能

  • 如果由malloc()函数分配的内存空间原来没有被使用过,则其中的每一位可能都是0;
  • 反之, 如果这部分内存曾经被分配过,则其中可能遗留有各种各样的数据.
  • 也就是说,使用malloc()函数的程序开始时(内存空间还没有被重新分配)能正常进行,但经过一段时间(内存空间已经被重新分配)那么就可能会出现问题.

   (2)函数calloc() 会将所分配的内存空间中的每一位都初始化为零

  • 也就是说,如果你是字符类型或整数类型的元素分配内存,那么这些元素将保证会被初始化为0;
  • 如果你是为指针类型的元素分配内存,那么这些元素通常会被初始化为空指针;
  • 如果你为实型数据分配内存,则这些元素会被初始化为浮点型的零.

  (3)函数malloc向系统申请分配指定size个字节的内存空间.返回类型是 void*类型.

  • void*表示未确定类型的指针.C,C++规定,void* 类型可以强制转换为任何其它类型的指针.

  (4)realloc可以对给定的指针所指的空间进行扩大或者缩小,无论是扩张或是缩小,原有内存的中内容将保持不变.

  • 当然,如果是缩小,则被缩小的那一部分的内容会丢失.realloc并不保证调整后的内存空间和原来的内存空间保持同一内存地址.
  • 相反,realloc返回的指针很可能指向一个新的地址.

   (5)realloc是从堆上分配内存的.

  • 当扩大一块内存空间时,realloc()试图直接从堆上现存的数据后面的那些字节中获得附加的字节,如果能够满足,自然天下太平;
  • 如果数据后面的字节不够,问题就出来了,那么就使用堆上第一个有足够大小的自由块,现存的数据然后就被拷贝至新的位置,而老块则放回到堆上.这句话传递的一个重要的信息就是数据可能被移动.

3. C++中动态内存管理

3.1 new/delete操作内置类型

C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力,而且使用起来比较麻烦,因 此C++又提出了自己的内存管理方式:

  • 通过new和delete操作符进行动态内存管理。

首先第一个知识点来了,那就是小伙伴们知道默认情况下【new】会不会初始化呢?

我通过调试带大家仔细瞧瞧是不是像我所说的那样:

  • 从上述我们不难发现,不管是对于【new】定义的【p1】来说,还是通过传统的【malloc】方式定义的【p2】来说,默认都是没有进行初始化操作的!!!

那么大家就会好奇了,【new】是否可以初始化呢?

  • 答案当然是可以的,接下来我带大家看看具体的操作。

此时,一个可能让大家混淆的点就出现了,大家是否能够区别下述这种方法呢?😖

int* p1 = new int(1);//初始化
int* p3 = new int[10];

大家是否知道这两个的区别呢?不知道没关系,接下来我给大家解答一下:

  • 对于【int* p1 = new int(1);】:这是进行初始化操作,申请一个(int)类型,并初始化为1;
  • 对于(int* p3 = new int[10];):它的意思动态申请10个int类型的空间
  • 大家一定区分二者之间的差别,不要搞混淆了!!

那对于(int* p3 = new int[10];)这种情况,我们是否还能对其初始化呢?其实也是可以的,C++支持这样的操作,具体如下所示:

int* p4 = new int[10] {1, 2, 3, 4};

当我们想对其进行初始化时,只需在后面加上【{}】即可,那么是不是呢?我通过调试给大家展示:

 

我们从上述不难看出,当我们用【new】时是不是比我们用【malloc】方便得多呀!

  • 对于【malloc】我们不仅需要强转类型,还需要进行检查

还是通过代码,大家就可以一目了然两者之间的差别到底有多大:

注意:

  1. 此时很多小伙伴或许就会问,难道【new】不会失败吗?其实是会的,它失败的机制跟【malloc】是不一样的,它通过“抛异常”来进行错误判别的,我们后面会讲到。

我用一张图给大家总结,大家看下表就能直观的理解上面的知识了:

特别注意一点:💣

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

3.2 new和delete操作自定义类型

上述我们已经知道一点,new/delete 和 malloc/free对于内置类型的处理几乎是一样的,但是对于自定义类型是否也是一样的呢?

我们还是通过代码来大家理解这个问题,大家看以下代码,最终的结果是什么呢?

class A
{
public:
  A(int a = 0)
    : _a(a)
  {
    cout << "A():" << this << endl;
  }
  ~A()
  {
    cout << "~A():" << this << endl;
  }
private:
  int _a;
};
int main()
{
  A* p1 = (A*)malloc(sizeof(A));
  A* p2 = new A(1);
  free(p1);
  delete p2;
    return 0;
}

解析:

我们直接打印看看最后的结果是什么。

  • 通过上述不知道大家有没有发现一个点呀!那就是此时这里我们是定义的两个,怎么只调用了一次构造和析构函数呢?是谁没有调用呢?接下来,通过调试,我带大家一步一步的去查看

解析:

  • 第一步,首先是对【p1】进行的操作,此时当执行完【p1】之后,我们发现并没有去调用构造函数。

解析:

  • 此时,当我们去执行完【p2】之后,我们发现,此时程序就去调用了构造函数,并且成功的完成了初始化操作。继续执行

  解析:

  • 当执行完毕,对其进行释放的时候,我们可以发现,程序是先对【p1】进行的释放,但是此时并没有去调用析构函数。

  解析:

  • 最后,当我们对【p2】进行释放的时候,此时程序调用了析构函数,并且也成功的把【p2】释放掉了。

因此,综上所述:在申请自定义类型的空间时,new会调用构造函数,delete会调用析构函数,而malloc与 free不会。


4. operator new与operator delete函数😎

  • 上述我们已经知道对于 【new】和【delete】是用户进行动态内存申请和释放的操作符;
  • 而接下来要学习的operator new 和operator delete是系统提供的全局函数;
  • new在底层调用operator new全局函数来申请空间,delete在底层通过 operator delete全局函数来释放空间。

大家是不是一看到这个就以为是函数重载呀!在之前我们已经学习过,【new】是函数,而【operator】是重载的符号:

  • 其实不然啊,大家千万不要这么理解。这里的两个函数是库里面提供的两个全局函数,不是运算符重载哟!!!
  • 取这个名字给大家造成了极大的误解,至于为什么要这么定义呢?我们也不得而知了,可能我们的祖师爷在设计的时候没有想到好名字,这就造成了许多学习【C++】的在这里吃了一个亏。

接下来带大家浅浅的看一下库里面是怎么实现这两个函数的,以下为库里面的代码(看不明白没关系)

对于【operator new】,库里面是这么写的:

/*
operator new:该函数实际通过malloc来申请空间,当malloc申请空间成功时直接返回;申请空间
失败,尝试执行空间不足应对措施,如果改应对措施用户设置了,则继续申请,否则抛异常。
*/
void *__CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc)
{
// try to allocate size bytes
void *p;
while ((p = malloc(size)) == 0)
  if (_callnewh(size) == 0)
     {
         // report no memory
         // 如果申请内存失败了,这里会抛出bad_alloc 类型异常
         static const std::bad_alloc nomem;
         _RAISE(nomem);
     }
return (p);
}

解析:

  • 大家看代码,我们可以看到operator new里面是不是调用的【malloc】啊,只是它这里跟【malloc】不同的是,紧接着往下看我们可以发现,上述代码【malloc】之后赋值给了【p】,然后判断,如果【p】等于0的话就调用【(_callnewh(size) == 0】这个函数,然后进入里面进行了“抛异常”,即【malloc】失败了就会抛异常。

而对于【operator delete】,库里面是这么写的:

/*
operator delete: 该函数最终是通过free来释放空间的
*/
void operator delete(void *pUserData)
{
     _CrtMemBlockHeader * pHead;
     RTCCALLBACK(_RTC_Free_hook, (pUserData, 0));
     if (pUserData == NULL)
         return;
     _mlock(_HEAP_LOCK);  /* block other threads */
     __TRY
         /* get a pointer to memory block header */
         pHead = pHdr(pUserData);
          /* verify block type */
         _ASSERTE(_BLOCK_TYPE_IS_VALID(pHead->nBlockUse));
         _free_dbg( pUserData, pHead->nBlockUse );
     __FINALLY
         _munlock(_HEAP_LOCK);  /* release other threads */
     __END_TRY_FINALLY
     return;
}
/*
free的实现
*/
#define   free(p)               _free_dbg(p, _NORMAL_BLOCK)

解析:

  • 上面那一大串的大家都可以不用看,看最后的一两行,最后是不是显示的【free】啊!

因此,我们可以得出一个结论:

  • 通过上述两个全局函数的实现知道,operator new 实际也是通过malloc来申请空间,如果 malloc申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供该措施就继续申请,否则就抛异常。
  • operator delete 最终是通过free来释放空间的。
  • 因此,【operator delete】和【operator new】本质上就是【delete】和【new】的封装。

接下来就给大家讲解其中的原因?

首先,我们怎么使用者两个函数呢?其实吧,这两个函数的实现跟【malloc】是类似的,我们举例观察:

解析:

  • 对于上述的【p1】,我们使用的是【operator new】,我们不难发现跟【malloc】的实现方式很类似。
  • 只是底层的机制不一样罢了。对于【malloc】实现,会进行严格的检查,而对于【operator new】则是失败后“抛异常”!!

紧接着大家是否好奇为什么会有这两个函数呢?

  • 我们都知道【c++】兼容C语言,就拿我们之前以及讲过的“引用”为例,当在【C++】中为了实现这个会去单独的创造一套语法啊什么的出来吗?应该不会吧。引用底层是按指针的方式来实现引用,那么这时还有必要再去创造新的东西呢?结果可能而知。
  • 现在我们来对比一下【new】的实现,第一步是申请空间紧接着就是调用构造函数对于申请空间是不是就是去堆上申请啊,在【C】语言中我们也是从堆上申请的呀!而在【C】语言中,对于申请空间,是不是就要调用【malloc】。
  • 而对于【C++】来说,因为它是面向对象的语言,虽然兼容C语言,但是新增的那部分是面向对象的,而面向对象的语言处理异常使用的是抛异常”。而在【C】语言中,对于失败,返回的是【null】,就不符合我们的需求,所以C++就用operator new去封装【malloc】,封装【malloc】失败之后“抛异常”!!!🥶

我们在通过反汇编的角度去看看,不难发现底层确实像我们所说的那样:


 

5. new和delete的实现原理

5.1 内置类型

  • 如果申请的是内置类型的空间,new和malloc,delete和free基本类似.
  • 不同的地方是: new/delete申请和释放的是单个元素的空间,new[]和delete[]申请的是连续空间,而且new在申 请空间失败时会抛异常,malloc会返回NULL

5.2 自定义类型

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来释 放空间

6. 定位new表达式(placement-new)

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

使用格式:

  • new (place_address) type或者new (place_address) type(initializer-list)
  • place_address必须是一个指针,initializer-list是类型的初始化列表

我们先看一下以下代码:

int main()
{
  A aa;
  A* p1 = (A*)malloc(sizeof(A));
  if (p1 == nullptr)
  {
    perror("malloc fail");
  }
  return 0;
}

解析:

此时,我们这里有一块已经【malloc】出来的空间,此时当我们在想对其进行初始化时是不行的

 

此时对于【C++】来说,对已经有一块空间的进行初始化,此时应该怎么办呢?基于这种情况,就引入了——定位new

new(p1)A;  // 注意:如果A类的构造函数有参数时,此处需要传参
new(p1)A(1);

接下来,我们通过调试带大家去看看:

 

解析:

  • 对于没有参数的时候,调试出来的我们发现也没有参数。

解析:

  • 对于有参数的时候,调试出来的我们对其进行了初始化操作。

当我们想去调用析构函数时,我们可以显示的去调用是可以的,具体如下:

int main()
{
  A aa;
  A* p1 = (A*)malloc(sizeof(A));
  if (p1 == nullptr)
  {
    perror("malloc fail");
  }
  //new(p1)A;  // 注意:如果A类的构造函数有参数时,此处需要传参
  new(p1)A(1);  
  p1->~A(); //调用析构函数,显示的去调用
  free(p1);
  return 0;
}

输出结果为:

 

大家看完这个是不是会觉得十分麻烦呀!这么复杂搞得,我们直接【new】不是更好吗?具体如下:

A* p2 = new A;
  • 其实确实是这样的,在实际当中对其自定义类型,我们直接用【new】就可以了,没必要再去【malloc】紧接着再去使用这个定位new。

那么何时我们改用这个呢?

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

如果还没明白。接下来我讲个故事,大家可以类比的进行理解:

  • 在以前,假如还没有水缸之类的东西用于存储,假如此时我们要煮饭,我们就去水井里舀水回来,总之就是当我们想使用水时时就要去水井里盛水回来;
  • 现在引入内存池,意思就相当于现在有装水的容器了,我们可以一次性盛许多水回来,当我们想用时就不用再每次去水井里盛水了,只有当装水的容器里没了之后,我们才去水井里盛水。

7. 常见面试题

7.1 malloc/free和new/delete的区别

malloc/free和new/delete的共同点是:

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

不同的地方是:

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

8.总结

以上便是关于C/C++内存管理的问题,还有部分关于内存泄漏的问题,我们到后面在讲。最后,总结一下本期的内容。

对于内存管理呢,C/C++其实是类似的,唯独在C++中引入了【new】和【delete】机制。

并且建议使用【new】和【delete】机制,原因有二:

  • 第一是因为用起来方便许多相比于【malloc】这种方式;
  • 第二个原因针对自定义类型,【new】和【delete】能更好的调用构造函数和析构函数,而【malloc】则不满足这种场景。

到此,便是关于内存管理的所有内容。如果感觉对您有帮助的话,麻烦点赞三连哟!!!

 

相关文章
|
9天前
|
存储 缓存 编译器
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
|
2月前
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
59 3
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
152 4
|
3月前
|
存储 程序员 编译器
简述 C、C++程序编译的内存分配情况
在C和C++程序编译过程中,内存被划分为几个区域进行分配:代码区存储常量和执行指令;全局/静态变量区存放全局变量及静态变量;栈区管理函数参数、局部变量等;堆区则用于动态分配内存,由程序员控制释放,共同支撑着程序运行时的数据存储与处理需求。
182 21
|
3月前
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
|
3月前
|
存储 C语言 C++
【C++打怪之路Lv6】-- 内存管理
【C++打怪之路Lv6】-- 内存管理
56 0
【C++打怪之路Lv6】-- 内存管理
|
3月前
|
存储 C语言 C++
【C/C++内存管理】——我与C++的不解之缘(六)
【C/C++内存管理】——我与C++的不解之缘(六)
|
3月前
|
程序员 C语言 C++
C++入门5——C/C++动态内存管理(new与delete)
C++入门5——C/C++动态内存管理(new与delete)
98 1
|
3月前
|
编译器 C语言 C++
详解C/C++动态内存函数(malloc、free、calloc、realloc)
详解C/C++动态内存函数(malloc、free、calloc、realloc)
475 1
|
3月前
|
存储 安全 程序员
【C++篇】深入内存迷宫:C/C++ 高效内存管理全揭秘
【C++篇】深入内存迷宫:C/C++ 高效内存管理全揭秘
117 3