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


相关文章
|
24天前
|
存储 缓存 编译器
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
|
5天前
|
存储 程序员 编译器
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
|
2月前
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
65 3
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
199 4
|
3月前
|
存储 程序员 编译器
简述 C、C++程序编译的内存分配情况
在C和C++程序编译过程中,内存被划分为几个区域进行分配:代码区存储常量和执行指令;全局/静态变量区存放全局变量及静态变量;栈区管理函数参数、局部变量等;堆区则用于动态分配内存,由程序员控制释放,共同支撑着程序运行时的数据存储与处理需求。
207 22
|
3月前
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
|
3月前
|
存储 C语言 C++
【C++打怪之路Lv6】-- 内存管理
【C++打怪之路Lv6】-- 内存管理
63 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)
104 1
|
3月前
|
编译器 C语言 C++
详解C/C++动态内存函数(malloc、free、calloc、realloc)
详解C/C++动态内存函数(malloc、free、calloc、realloc)
594 1