new后未delete导致泄漏;2. 数组用delete而非delete[]引发未定义行为;3. 异常使释放代码跳过,需RAII;4. 类成员未在析构函数释放,应遵循三/五法则;5. 指针重赋值前未释放原内存;6. 容器存指针不清理,应遍历delete或用智能指针。推荐智能指针和RAII,辅以Valgrind等工具检测。

C++中内存管理是开发者必须掌握的核心技能之一。由于缺乏自动垃圾回收机制,程序员需要手动管理动态分配的内存,稍有不慎就可能导致内存泄漏。以下是初学者常遇到的几种典型内存泄漏场景,了解它们有助于写出更安全的代码。
1. 动态内存分配后未释放
使用 new 分配内存后,如果没有对应的 delete,就会造成泄漏。
例如:
int* ptr = new int(10);// 忘记 delete ptr;
// 程序结束前未释放,内存泄漏
建议:每次使用 new 后,确保在适当位置调用 delete。更推荐使用智能指针替代裸指针。
立即学习“C++免费学习笔记(深入)”;
2. 数组分配后使用 delete 而非 delete[]
用 new[] 分配数组时,必须用 delete[] 释放,否则行为未定义,可能引发资源泄漏。
char* buffer = new char[100];delete buffer; // 错误!应使用 delete[]
// 可能只释放第一个元素,其余内存未正确回收
说明:delete 和 delete[] 的内部机制不同,混用会导致析构不完整。
3. 异常导致的提前退出
在分配内存后,如果发生异常而跳过释放语句,内存将无法回收。
int* ptr = new int[50];someFunctionThatMightThrow(); // 抛出异常
delete[] ptr; // 不会被执行
解决方法:使用 RAII 技术,比如 unique_ptr 或直接使用 vector 等容器,让资源在异常时自动释放。
4. 忘记释放类中动态分配的成员
在类中使用指针成员并用 new 初始化时,如果没有在析构函数中释放,实例销毁时会造成泄漏。
class MyClass {int* data;
public:
MyClass() { data = new int[100]; }
// 没有定义析构函数
};
// 每次创建对象都会泄漏 100 个 int 的内存
建议:遵循“三法则”或“五法则”,当类管理资源时,显式定义析构函数、拷贝构造函数和赋值操作符,或直接使用智能指针托管资源。
5. 指针被重新赋值前未释放原内存
指向动态内存的指针被直接赋新值,导致原内存地址丢失,无法释放。
int* ptr = new int(5);ptr = new int(10); // 原内存未释放,直接丢失指针
说明:第一次分配的内存再也无法访问,形成泄漏。应先释放再分配:
delete ptr;ptr = new int(10);
6. 容器中存储指针但未清理
使用 vector 等容器保存原始指针时,容器销毁不会自动释放指针指向的内存。
std::vectorvec.push_back(new int(1));
vec.push_back(new int(2));
// vec.clear() 或析构时,int 内存不会被释放
建议:遍历容器并逐一 delete,或改用智能指针如 std::vector<:unique_ptr>>。
基本上就这些。养成使用智能指针、遵循 RAII 原则的习惯,能大幅减少内存泄漏风险。工具如 Valgrind 或 AddressSanitizer 也能帮助检测泄漏问题。










