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


目录
打赏
0
0
0
0
0
分享
相关文章
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
1月前
|
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
23 1
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
298 68
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
253 0
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
167 0
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
140 3
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
484 4
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
|
9月前
|
【C++打怪之路Lv6】-- 内存管理
【C++打怪之路Lv6】-- 内存管理
94 0
【C++打怪之路Lv6】-- 内存管理
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问