首页 > 后端开发 > C++ > 正文

C++内存泄漏场景 常见案例与分析

P粉602998670
发布: 2025-08-27 09:43:01
原创
1054人浏览过
C++内存泄漏主因是动态内存未释放,常见场景包括:1. new后未delete;2. new[]未用delete[];3. 异常导致delete被跳过;4. 指针丢失;5. 类析构函数未释放成员;6. shared_ptr循环引用;7. 资源未关闭。应使用智能指针、RAII和检测工具防范。

c++内存泄漏场景 常见案例与分析

C++内存泄漏通常发生在动态分配的内存没有被正确释放的情况下。由于C++不提供自动垃圾回收机制,开发者必须手动管理内存,稍有疏忽就可能导致泄漏。以下是几种常见的内存泄漏场景及其分析,帮助识别和避免这类问题。

1. new 之后未 delete

这是最基础也最常见的内存泄漏场景。

使用 new 分配内存后,若未在适当位置调用 delete,内存将一直被占用。

示例:

void leakExample() {
    int* ptr = new int(10);
    // 忘记 delete ptr;
}
登录后复制

每次调用该函数都会泄漏一个 int 大小的内存。解决方法是在使用完后显式释放:

立即学习C++免费学习笔记(深入)”;

delete ptr;

2. 数组使用 new[] 但未用 delete[]

使用 new[] 分配数组时,必须用 delete[] 释放,否则行为未定义,且可能造成资源泄漏。

示例:

void arrayLeak() {
    char* buffer = new char[100];
    // delete buffer;  // 错误:应使用 delete[]
}
登录后复制

正确做法是:

delete[] buffer;

混用 deletedelete[] 可能导致析构不完整或内存管理器损坏。

3. 异常导致的提前退出

在分配内存后,如果发生异常,程序可能跳过 delete 语句。

示例:

void exceptionLeak() {
    Resource* res = new Resource();
    res->initialize(); // 可能抛出异常
    delete res;
}
登录后复制

initialize() 抛出异常,delete res 不会被执行。

解决方案是使用 RAII(Resource Acquisition Is Initialization)技术,例如智能指针:

存了个图
存了个图

视频图片解析/字幕/剪辑,视频高清保存/图片源图提取

存了个图 17
查看详情 存了个图

void safeException() {
    std::unique_ptr<Resource> res = std::make_unique<Resource>();
    res->initialize(); // 异常发生时,unique_ptr 自动释放资源
}
登录后复制

4. 指针被覆盖或丢失

当指向动态内存的指针被重新赋值或作用域结束而未释放,内存将无法访问,造成泄漏。

示例:

void lostPointer() {
    int* ptr = new int(5);
    ptr = new int(10); // 原内存地址丢失,泄漏第一个 int
}
登录后复制

第一次分配的内存没有被释放就被覆盖。应先释放再重新分配:

int* ptr = new int(5);
delete ptr;
ptr = new int(10);
登录后复制

5. 忘记释放类中的动态成员

类中使用指针成员时,若未在析构函数中释放,会造成实例销毁时内存泄漏。

示例:

class Data {
    int* buffer;
public:
    Data() { buffer = new int[100]; }
    ~Data() { /* 未释放 buffer */ }
};
登录后复制

每次创建 Data 对象都会泄漏 100 个 int 的内存。应补充析构函数:

~Data() { delete[] buffer; }

同时注意遵循“三法则”或“五法则”,正确实现拷贝构造、赋值操作等。

6. 循环引用导致智能指针无法释放

使用 std::shared_ptr 时,若出现循环引用,引用计数永不归零,内存无法释放。

示例:

struct Node {
    std::shared_ptr<Node> parent;
    std::shared_ptr<Node> child;
};
// A -> B, B -> A,形成循环,无法释放
登录后复制

应使用 std::weak_ptr 打破循环:

将 parent 或 child 中的一个改为 std::weak_ptr,避免计数循环。

7. 未关闭文件句柄或资源(广义泄漏)

虽然不是传统意义上的内存泄漏,但未释放文件、套接字等系统资源,也会导致资源耗尽。

应使用 RAII 包装资源,如 std::fstream 自动关闭文件,或自定义析构函数释放资源。

基本上就这些常见场景。关键是养成使用智能指针、RAII 和异常安全编程的习惯,配合工具如 Valgrind、AddressSanitizer 检测泄漏,能大幅降低风险。

以上就是C++内存泄漏场景 常见案例与分析的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号