本文介绍: 堆内存指的是程序执行中依据须要分配通过malloc / calloc / realloc / new等从堆中分配的一块内存,用完后必须通过调用相应的 free或者delete 删掉可以重载全局operator newoperator delete函数,但是不能重载newdelete运算符,注意“operator new”是一个函数名称,而不是对new进行了重载的意思;内存泄漏并不是指内存物理上的消失,而是应用程序分配某段内存后,因为设计错误,失去了对该段内存控制,因而造成了内存浪费

newdelet 概念

newdelete用于动态内存管理运算符用于在堆上分配和释放内存new运算符用于在堆上分配内存,并调对象构造函数进行初始化

new 和 delet使用

注意

  1. 括号是给类型构造函数传参,方括号号是确定动态开辟该类型空间大小
  2. new操作符开辟的空间必须由delet释放,new[]开辟的空间必须由delet[]释放 。
  3. new/deletemalloc/free最大区别是 new/delete对于【自定义类型】除了开空间还会调用构造函数和析构函数
    在这里插入图片描述

new与 delete 底层原理

  1. new和delete操作符通过调用 operator new与operator delete 全局函数实现功能

  2. 可以重载全局operator new和operator delete函数,但是不能重载new和delete运算符,注意“operator new”是一个函数名称,而不是对new进行了重载的意思 ;

  3. operator new和operator delete函数内部逻辑大致遵从以下形式/逻辑

cpp
void* operator new(std::size_t size)
{
    void* ptr = std::malloc(size); // 使用标准库函数 malloc 分配内存
    if (ptr == nullptr) // 内存分配失败
    {
        throw std::bad_alloc(); // 抛出 bad_alloc 异常
    }
    return ptr;
}

void operator delete(void* ptr)
{
    // 自定义内存释放逻辑
    // ...
    free(ptr);
}

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在释放空间前会调用析构函数完成空间中资源清理

new / opera new / 构造函数 之间关系

new[] / delete[] = operator new[] / operator delete[] + 多次构造函数

operator new[] / operator delete[] = 多次调用operator delete / operator new

operator new 成功:
operator new = malloc函数 / 自定义重载操作 + malloc函数
operator new 失败
operator new = 抛异常)/ 抛异常 + 自定义重载操作

operator delet = free函数 / 自定义重载操作 + free函数

new = operator new 函数 + 构造函数 (注意先后顺序)

delete = 析构函数 + operator delet 函数 (注意先后顺序)

  1. new / new[] 失败时会抛异常

定位new表达式(placement-new)

new (pointer) Type [initializer];

pointer一个指向已分配内存的指针,Type是要构造对象类型,initializer是可选的初始化列表

定位new表达式(placement new)是一种特殊的new表达式用于在已分配的内存地址构造对象。它允许我们指定的内存位置创建对象,而不是在堆上动态分配内存。

使用

class A
{
public:
 A(int a = 0)
 : _a(a)
 {
 cout << "A():" << this << endl;
 }
 ~A()
 {
 cout << "~A():" << this << endl;
 }
private:
 int _a;
};


int main()
{
 /* p1现在指向的只不过是与A对象相同大小的一段空间,
 还不能算是一个对象,因为构造函数没有执行*/
 A* p1 = (A*)malloc(sizeof(A));
 
 // !!!!!!!!!!!!!!!!!!!!!!
 // 定位new/replacement new
 // 注意:如果A类的构造函数有参数时,此处需要传参
 new(p1)A;  
 p1-&gt;~A();
 free(p1);
 
 // !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! 
 A* p2 = (A*)operator new(sizeof(A));
 new(p2)A(10);
 p2-&gt;~A();
 
 operator delete(p2);
 
 return 0;
}

内存泄漏

内存泄漏指因为疏忽或错误造成程序未能释放已经不再使用的内存的情况。内存泄漏并不是指内存在物理上的消失,而是应用程序分配某段内存后,因为设计错误,失去了对该段内存的控制,因而造成了内存的浪费
内存泄漏的危害:长期运行程序出现内存泄漏影响很大,如操作系统后台服务等等,出现内存泄漏会导致响应越来越慢,最终卡死

内存泄漏分类

堆内存泄漏(Heap leak)
堆内存指的是程序执行中依据须要分配通过malloc / calloc / realloc / new等从堆中分配的一块内存,用完后必须通过调用相应的 free或者delete 删掉假设程序设计错误导致这部分内存没有被释放,那么以后这部分空间将无法再被使用,就会产生Heap Leak
系统资源泄漏
指程序使用系统分配的资源,比方套接字文件描述符管道没有使用对应的函数释放掉,导致系统资源的浪费,严重可导致系统效能减少,系统执行稳定

如何对待内存泄漏

  1. 有良好的编程习惯
  2. 使用内存泄漏检测工具:有一些专门的工具可以帮助检测内存泄漏,例如Valgrind、AddressSanitizer(ASan)、LeakSanitizer(LSan)等。这些工具可以运行检测程序中的内存泄漏,并给出相应的报告调试信息
  3. 重载 operator new 和 operator delete:通过重载全局的 operator new 和 operator delete 函数,可以跟踪内存的分配和释放情况。可以在这些函数中记录分配的内存块信息,并在程序结束检查是否有未释放的内存块。
  4. 使用智能指针:使用智能指针(如std::shared_ptrstd::unique_ptr)可以自动管理内存的释放,避免手动释放内存的疏忽。智能指针使用引用计数独占所有权方式管理资源,确保在不再需要时正确释放内存。
  5. 编写测试用例编写全面的测试用例覆盖程序的各个功能模块代码路径,以确保内存的正确分配和释放。通过测试用例执行结果检查是否有内存泄漏的情况。
  6. 代码审查和静态分析工具:进行代码审查,特别关注内存分配和释放的逻辑。同时使用静态分析工具(如Clang静态分析器、Cppcheck等)来检查代码潜在的内存泄漏问题

注 : 内存泄漏检测工具只是有概率查处错误 。

原文地址:https://blog.csdn.net/WSK1454360679/article/details/134746052

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任

如若转载,请注明出处:http://www.7code.cn/show_33094.html

如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱suwngjj01@126.com进行投诉反馈,一经查实,立即删除

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注