c++内存管理

简介: c++内存管理

文章目录

new和delete

new和malloc的区别&delete和free的区别

new动态开辟空间还可以调用其构造函数对其初始化,可以对于之定义类型进行初始化,

malloc只会动态开辟空间,不会初始化

但是对于内置类型没有区别


delete可以完成资源的清理和空间的销毁,对于自定义类型可以调用其析构函数完成其资源的清理

free只会完成空间的销毁


用法:

class A
{
  int _a;
  int _b;
public:
  A()
  {
    cout << "A()" << endl;
  }
  ~A()
  {
    cout << "~A()" << endl;
  }
};
int main()
{
  //总结malloc/free和new/delete 对于内置类型没有本质的区别,只有用法上的区别
  //动态申请int和5个int的数组
  //c语言
  int* ptr1 = (int *)malloc(sizeof(int));
  int* ptr2 = (int*)malloc(sizeof(int) * 5);
  //c++
  int* p3 = new int;//动态开辟一个int
  int* p4 = new int[5];//动态申请5个int的空间
  int* p5 = new int(5);//申请1个int空间,**初始化**为5
  int* p6 = new int[5]{ 1,2 };//5个int也可以这样初始化
  //删除
  free(ptr1);
  free(ptr2);
  ptr1 = nullptr;
  ptr2 = nullptr;
  delete p3;
  delete[] p4;
  p3 = nullptr;
  p4 = nullptr;
  A* pa = (A*)malloc(sizeof(A));
  A* pa2 = new A;
  //对于自定义类型,new还可以调用其初始化,还可以开空间
  //malloc只会开空间
  A* pa3 = (A*)malloc(sizeof(A) * 5);
  A* pa4 = new A[5];
  //delete要先调用指针类型的析构函数,再去释放空间给堆上
  delete pa3;
  delete[] pa4;
  return 0;
}

需要注意的是new要和delete配对

new[]要和delete[]配对

new和delete的应用

class Stack
{
private:
  int _top;
  int _capacity;
  int* _a;
public:
  Stack(int capacity = 4)
    :_top(0)
    ,_capacity(4)
  {
    _a = new int[capacity];//对于*a的处理初始化非常方便
  }
  ~Stack()
  {
    delete[] _a;//清理资源
    _a = nullptr;
  }
};
int main()
{
  Stack* s1 = new Stack;//会自动调用构造函数和开空间,
  delete s1;//调用析构函数,清理对象中的资源再释放空间,先把里面的资源给干掉,再释放掉s1指向的这空间给释放掉
return 0;
}
```cpp
struct ListNode
{
  ListNode* prev;
  ListNode* next;
  int _val;
  ListNode(int val)//初始化列表
    :prev(nullptr)
    ,next(nullptr)
    ,_val(val)
  {}
};
class List
{
public:
  List()//双向带头循环链表,构造函数初始化
  {
    _head = new ListNode(-1);
    _head->next = _head;
    _head->prev = _head;
  }
  void pushback(int val)
  {
    ListNode* newnode = new ListNode(val);//这样用的话就是用一次就删除一次,因为new调用operator new ,operator new又调用malloc,
    //所以我们就可以存在一个新的operator new,使用内存池
    ListNode* tail = _head->prev;
    tail->next = newnode;
    newnode->prev = tail;
    _head->prev = newnode;
    newnode->next = _head;
  }
private:
  ListNode* _head;
};
int main()
{
struct ListNode* n1 = (struct ListNode*)malloc(sizeof(ListNode));
  n1->prev = nullptr;
  n1->next = nullptr;
  n1->_val = 0;
  //cpp
  ListNode* n2 = new ListNode(0);//更加容易
return 0;
}

new和delete的底层

operator new和operator delete

operator new中调用malloc申请内存,但是malloc失败之后返回null,而operator new失败以后,改为抛异常处理错误,这样符合c++面向对象语言处理错误的方式

这个operator new是给new用的一般不是给我们用的


new就是使用operator new(申请空间)+调用构造函数(初始化)

Stack* ps1 = (Stack*)operator new(sizeof(Stack));//只会开空间不会调用构造函数,

同理operator delete就是只会申请空间,不会调用析构函数,也是给delete所使用的

operator delete ps1;//就只会销毁空间,和free差不多

定位new

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

用法:

new(地址)类型(值)


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

class A
{
public:
  A(int a)
    :_a(a)
  {}
  ~A()
  {
  }
private:
  int _a;
};
int main()
{
  A* p = (A*)malloc(sizeof(A));
  new(p)A(1);//new(地址)类型(值),对一块已经分配好的内存调用初始化构造函数,不开辟空间
  //对于内存池来的就用定位new
  //析构函数
  p->~A();//
  operator delete(p);
  return 0;
}

内存泄漏

1–动态申请的内存,不使用了,又没有主动释放,就存在内存泄漏了

2–内存泄露的危害:

a,出现内存泄露的进程会正常结束,进程结束时这些内存会还给系统,不会又危害

b,出现内存泄露的进程非正常结束,比如僵尸内存,系统用的内存越来越少

c。需要长期运行的程序出现内存泄露,危害很大,系统会越来越满,甚至卡死宕机 —服务器程序,


相关文章
|
14天前
|
存储 编译器 C语言
内存管理【C++】
内存管理【C++】
33 1
|
28天前
|
存储 编译器 C语言
【C++】C\C++内存管理
【C++】C\C++内存管理
【C++】C\C++内存管理
|
27天前
|
编译器 C++
virtual类的使用方法问题之C++类中的非静态数据成员是进行内存对齐的如何解决
virtual类的使用方法问题之C++类中的非静态数据成员是进行内存对齐的如何解决
|
8天前
|
C语言 C++
C++(二)内存管理
本文档详细介绍了C++中的内存管理机制,特别是`new`和`delete`关键字的使用方法。首先通过示例代码展示了如何使用`new`和`delete`进行单个变量和数组的内存分配与释放。接着讨论了内存申请失败时的处理方式,包括直接抛出异常、使用`try/catch`捕获异常、设置`set_new_handler`函数以及不抛出异常的处理方式。通过这些方法,可以有效避免内存泄漏和多重释放的问题。
|
29天前
|
存储 Java C语言
【C++】C/C++内存管理
【C++】C/C++内存管理
|
1月前
|
存储 编译器 C语言
C++内存管理(区别C语言)深度对比
C++内存管理(区别C语言)深度对比
58 5
|
22天前
|
存储 程序员 编译器
c++学习笔记08 内存分区、new和delete的用法
C++内存管理的学习笔记08,介绍了内存分区的概念,包括代码区、全局区、堆区和栈区,以及如何在堆区使用`new`和`delete`进行内存分配和释放。
35 0
|
2月前
|
NoSQL Redis C++
c++开发redis module问题之在复杂的Redis模块中,特别是使用第三方库或C++开发时,接管内存统计有哪些困难
c++开发redis module问题之在复杂的Redis模块中,特别是使用第三方库或C++开发时,接管内存统计有哪些困难
|
2月前
|
存储 C语言 C++
【C/C++】动态内存管理( C++:new,delete)
C++的`new`和`delete`用于动态内存管理,分配和释放内存。`new`分配内存并调用构造函数,`delete`释放内存并调用析构函数。`new[]`和`delete[]`分别用于数组分配和释放。不正确匹配可能导致内存泄漏。内置类型分配时不初始化,自定义类型则调用构造/析构。`operator new`和`operator delete`是系统底层的内存管理函数,封装了`malloc`和`free`。定位`new`允许在已分配内存上构造对象,常用于内存池。智能指针等现代C++特性能进一步帮助管理内存。
|
2月前
|
存储 编译器 程序员
【C/C++】动态内存管理(C:malloc,realloc,calloc,free)
探索C++与C语言的动态内存管理:从malloc到new/delete,了解内存分布及栈、堆的区别。文章涵盖malloc、realloc、calloc与free在C中的使用,强调内存泄漏的风险。C++引入new和delete,支持对象构造与析构,还包括operator new和placement-new。深入分析内存管理机制,揭示C与C++在内存处理上的异同。别忘了,正确释放内存至关重要!