new/delete, malloc/free 内存泄漏如何检测

区别:

  1. 首先new/delete是运算符,malloc/free是库函数。
  2. malloc/free只开辟内存不初始化;new/delete及开辟内存也初始化。
  3. 抛出异常的方式:new/delete开辟失败使用抛出bad_alloc;malloc/free通过返回值判断。
  4. malloc和new区别:malloc是c语言中一个库幻术函数,按字节为数据分配内存,返回类型是    ‘ void * ’。因为他不知道分配的内存会被用于什么类型的对象。 new是运算符,需要传入类型,new相当于运算符的重载函数 operator new ->返回值自动转成指定的类指针 int*
  5. free不管是释放单个内存还是数组内存都是函数的调用,传入内存的首地址即可,而delete在删除数组时需要加一个[].

有几种类型的new:

  •         int *p1 = new int (20) ;
  •         int *p2 = new (nothrow) int ;
  •         const int *p3 = new const int(40);
  •         int data = 0; int *p4 = new (&data) int (50);  指定内存地址

C++中,如何设计一个程序检测内存泄漏问题?

  • 内存泄漏就是new操作没有对应的delete,我们可以在全局重写上面这些函数,在new操作里面用映射表记录都有哪些内存被开辟过,delete的时候把相应的内存资源删除掉,new和delete都有对应关系
    #include <iostream>
    #include <unordered_map>
    #include <mutex>
    
    std::unordered_map<void*, std::size_t> allocationMap;
    std::mutex allocMutex;
    
    void* operator new(std::size_t size) {
        std::lock_guard<std::mutex> lock(allocMutex);
        void* ptr = std::malloc(size);
        if (ptr == nullptr) {
            throw std::bad_alloc();
        }
        allocationMap[ptr] = size;
        return ptr;
    }
    
    void operator delete(void* ptr) noexcept {
        std::lock_guard<std::mutex> lock(allocMutex);
        auto it = allocationMap.find(ptr);
        if (it != allocationMap.end()) {
            allocationMap.erase(it);
        }
        std::free(ptr);
    }
    
  • 如果整个系统运行完了,我们发现,映射表记录的一些内存还没有被释放,就存在内存泄漏了!
    void checkForMemoryLeaks() {
        std::lock_guard<std::mutex> lock(allocMutex);
        if (!allocationMap.empty()) {
            std::cout << "Memory leaks detected:\n";
            for (auto& pair : allocationMap) {
                std::cout << "Address: " << pair.first << ", Size: " << pair.second << " bytes\n";
            }
        } else {
            std::cout << "No memory leaks detected.\n";
        }
    }
    

  • 我们用我们自定义的new和delete重载函数 接管整个应用的所有内存管理 ,对内存的开辟和释放都记录;也可以通过编译器既定的宏和API接口,把函数调用堆栈打印出来,到底在哪个源代码的哪一页的哪一行做了new操作没有delete
  • 除了重载 newdelete,还有一些现成的工具和库,如 Valgrind、AddressSanitizer 等,这些工具可以自动检测内存泄漏,而无需修改源代码。
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
mallocnew是用于动态分配内存的两种方法。 malloc是C语言中的函数,用于在堆上动态分配内存。它的函数原型是void *malloc(size_t size),返回一个指向分配内存的指针。分配的内存大小由size参数指定。malloc分配的内存位于堆上,可以使用free函数释放。 new是C++中的关键字,用于在自由存储区(可以是堆或静态存储区)动态分配内存new的使用方式有两种:newnew\[\]。new用于分配单个对象的内存,而new\[\]用于分配数组的内存new返回一个指向分配内存的指针。分配的内存大小由对象的类型决定。new分配的内存可以使用delete来释放,而new\[\]分配的内存应使用delete\[\]来释放。 虽然newmalloc都可以用于动态分配内存,但它们之间有一些区别。首先,new是C++的关键字,而malloc是C语言的函数。其次,new分配的内存位于自由存储区,而malloc分配的内存位于堆上。此外,newdelete是操作符,可以重载,而mallocfree是函数,不能重载。最后,由于newmalloc使用不同的内存管理机制,所以不能混合使用。也就是说,不能使用malloc分配的内存使用delete释放,反之亦然。 引用\[1\]解释了为什么new\[\]分配的空间用free()释放会出错,因为new\[\]分配空间返回的地址并不是它里面malloc分配空间的首地址,系统预留了sizeof(int)个字节。引用\[2\]说明了new分配的内存空间所在位置是自由存储区,而malloc在堆上动态分配内存。引用\[3\]提到了operator new /operator delete的实现可以基于malloc,而malloc的实现不可以去调用new。 #### 引用[.reference_title] - *1* [C++ malloc/free/new/delete详解(内存管理)](https://blog.csdn.net/TABE_/article/details/122179176)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [newmalloc的区别](https://blog.csdn.net/Dr_Cassie/article/details/96494444)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值