当前位置: 首页 > news >正文

C++常见内存泄漏案例分析以及解决方案

C++ 常见内存泄漏案例分析以及解决方案

1. 分配与释放不匹配

在动态内存管理中,使用new操作符分配的内存必须通过delete操作符显式释放。若未遵循这一规则,将导致内存泄漏。例如:

int *p = new int;
p = new int; // 错误:未释放先前分配的内存
delete p; // 仅释放最后一次分配的内存

对于使用new[]分配的数组,必须使用delete[]进行释放,否则同样会导致内存泄漏:

int* ptr = new int[2];
// 使用ptr...
delete[] ptr; // 正确:使用delete[]释放数组

类似地,使用malloc分配的内存必须通过free释放,否则也会造成内存泄漏:

int *p = (int*)malloc(sizeof(int));
p = (int*)malloc(sizeof(int)); // 错误:未释放先前分配的内存
free(p); // 正确:释放内存

此外,库函数如strdup()返回的内存需要显式释放,否则也会造成内存泄漏:

char* dup_str = strdup("example");
free(dup_str); // 释放内存,避免内存泄漏

2. 嵌套指针的不完全释放

在处理嵌套指针时,必须确保释放每个层级的内存。仅释放外层内存而不释放内层指针会导致内存泄漏:

int **b = new int*[M];
for(int i = 0; i < M; i++) {b[i] = new int[N];
}
// 释放内存
for(int i = 0; i < M; i++) {delete[] b[i];
}
delete[] b;

3. 基类析构函数非虚

在多态情况下,基类的析构函数必须是虚函数,以确保通过基类指针删除派生类对象时,能够正确调用派生类的析构函数,避免内存泄漏:

class A {
public:virtual ~A() {} // 正确:虚析构函数
};class B : public A {
public:~B() {}
};

4. 使用free不触发析构

使用malloc分配的类对象不会触发析构函数,导致内存泄漏:

MyClass* obj = (MyClass*)malloc(sizeof(MyClass));
new(obj) MyClass(44);
free(obj); // 错误:不会调用析构函数

正确的做法是使用new和delete:

MyClass* obj = new MyClass(44);
delete obj; // 正确:调用析构函数

5. 更新未释放内存的指针

在更新指针指向的内存时,必须先释放旧内存,否则会造成内存泄漏:

class Student {
private:char* mName;
public:// 构造函数和析构函数...void setName(const char* name) {if (this->mName != nullptr) {delete[] this->mName;}// 分配新内存...}
};

6. 无法删除引用指向的内存地址

返回动态分配内存的引用时,无法释放内存,导致内存泄漏:

int& allocateInteger() {int* p = new int(42);return *p;
}

正确的做法是返回指针:

int* allocateInteger() {int* p = new int(42);return p;
}

7. 循环引用

对象之间的循环引用会导致内存泄漏,尤其是在使用智能指针如shared_ptr时。使用weak_ptr可以解决这个问题:

class ClassA {
public:void setInnerPtr(weak_ptr<ClassB> pB) {p = pB;}
private:weak_ptr<ClassB> p;
};class ClassB {
public:void setInnerPtr(weak_ptr<ClassA> pA) {p = pA;}
private:weak_ptr<ClassA> p;
};

8. 野指针和悬挂指针

野指针指向未知或随机地址,而悬挂指针指向已释放的内存,两者都可能导致未定义行为:

int* p1; // 野指针
int* p2 = NULL; // 非野指针
p1 = new int(10); // 非野指针
delete p1; // 悬挂指针

9. 浅拷贝产生悬挂指针

未重写拷贝构造函数和赋值运算符的类,可能导致浅拷贝,从而产生悬挂指针:

class Student {
private:char* mName;
public:// 构造函数、析构函数、拷贝构造函数和赋值运算符...
};

10. 异常安全问题

在异常处理中,如果资源未在所有退出路径上正确释放,可能导致内存泄漏:

try {int* myData = new int[100];// 可能抛出异常delete[] myData;
} catch (...) {// 异常处理
}

11. 隐式内存泄漏

隐式内存泄漏包括内存碎片、内存管理块未归还操作系统、STL内部内存管理策略等,这些都可能导致内存耗尽:

  • 内存碎片:大量小内存块导致无法重新分配。
  • 内存管理块:运行时库可能不归还内存给操作系统。
  • STL内存管理:内存可能不会立即归还操作系统。
http://www.lryc.cn/news/506692.html

相关文章:

  • [LeetCode-Python版]206. 反转链表(迭代+递归两种解法)
  • 70 mysql 中事务的隔离级别
  • C语言二叉树
  • 智能工厂的设计软件 三种处理单元(NPU/GPU/CPU)及其在深度学习框架中的作用 之1
  • iOS swift开发系列--如何给swiftui内容视图添加背景图片显示
  • jmeter后端监视器
  • 服务器数据恢复—RAIDZ离线硬盘数超过热备盘数导致阵列崩溃的数据恢复案例
  • 面试题整理4----lvs,nginx,haproxy区别和使用场景
  • iOS - 超好用的隐私清单修复脚本(持续更新)
  • html <a>设置发送邮件链接、打电话链接 <a href=“mailto:></a> <a href=“tel:></a>
  • clickhouse-副本和分片
  • 2009 ~ 2019 年 408【计算机网络】大题解析
  • vue2使用render,js中写html
  • 如何有效划分服务器磁盘空间?具体的步骤和流程
  • labelme标签批量转换数据集json_to_dataset
  • Fisco-Bcos-java-SDK 利用java与fisco-Bcos区块链上的智能合约交互(以HelloWorld为例)
  • OpenHarmony-3.HDF Display子系统(6)
  • Nginx中Server块配置的详细解析
  • php学习资料分享
  • EE308FZ_Sixth Assignment_Beta Sprint_Sprint Essay 3
  • Eureka学习笔记-服务端
  • 无限次使用 cursor pro
  • 网站运维之整站同步
  • 【机器人】Graspness 端到端 抓取点估计 | 论文解读
  • 力扣2300.咒语和药水的成功对数(二分法)
  • WEB开发: 全栈工程师起步 - Python Flask +SQLite的管理系统实现
  • 云原生周刊:Kubernetes v1.32 正式发布
  • 京准电钟:电厂自控NTP时间同步服务器技术方案
  • 深入探索Flink的复杂事件处理CEP
  • clickhouse-数据库引擎