在C++中,内存泄漏通常与手动管理内存有关,而不是直接由引用计数引起,因为C++标准库本身并不提供自动的引用计数功能。但是,我们可以通过一个例子来间接说明引用(或指针)管理不当如何导致内存泄漏,尤其是当涉及复杂对象结构和所有权关系时,这种管理不当往往体现在循环引用上。
在C++中,如果一个对象分配在堆上(通过new
操作符),则必须手动使用delete
来释放其内存。忘记释放分配的内存就会导致内存泄漏。
假设我们有两个类,ClassA
和ClassB
,它们互相持有对方的指针,形成了一个循环引用的情况
1class ClassA {
2public:
3 ClassB* b;
4 ClassA(ClassB* _b) : b(_b) {}
5 ~ClassA() { delete b; } // 这里试图解决循环引用,但会导致错误的释放或双删
6};
7
8class ClassB {
9public:
10 ClassA* a;
11 ClassB(ClassA* _a) : a(_a) {}
12 ~ClassB() { delete a; } // 同样,这里的问题也是错误的释放或双删
13};
14
15int main() {
16 ClassA* a = new ClassA(nullptr);
17 ClassB* b = new ClassB(a);
18 a->b = b; // 形成循环引用
19
20 // 此时,即使main函数结束,a和b也不会被释放,因为它们互相引用,且没有地方去删除它们