区别:
- 首先new/delete是运算符,malloc/free是库函数。
- malloc/free只开辟内存不初始化;new/delete及开辟内存也初始化。
- 抛出异常的方式:new/delete开辟失败使用抛出bad_alloc;malloc/free通过返回值判断。
- malloc和new区别:malloc是c语言中一个库幻术函数,按字节为数据分配内存,返回类型是 ‘ void * ’。因为他不知道分配的内存会被用于什么类型的对象。 new是运算符,需要传入类型,new相当于运算符的重载函数 operator new ->返回值自动转成指定的类指针 int*
- 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
- 除了重载
new
和delete
,还有一些现成的工具和库,如 Valgrind、AddressSanitizer 等,这些工具可以自动检测内存泄漏,而无需修改源代码。