【C++】weak_ptr、循环引用与线程安全

shared_ptr解决了共享所有权的问题,但引入了一个新坑:循环引用。当两个对象互相持有对方的shared_ptr,引用计数永远降不到零,资源就永远不会释放。weak_ptr正是为此而生。

目录

[1. 循环引用:图说明问题](#1. 循环引用:图说明问题)

[2. weak_ptr:打破循环](#2. weak_ptr:打破循环)

[3. 线程安全:引用计数要原子](#3. 线程安全:引用计数要原子)

[4. shared_ptr的另几个常用能力](#4. shared_ptr的另几个常用能力)

[5. 内存泄漏的最后一道防线](#5. 内存泄漏的最后一道防线)


1. 循环引用:图说明问题

常见场景:双向链表节点、树节点中存父子指针、观察者模式中Subject和Observer互指。

cpp

复制代码
struct ListNode {
    int _data;
    shared_ptr<ListNode> _next;
    shared_ptr<ListNode> _prev;
    ~ListNode() { cout << "~ListNode()" << endl; }
};

int main() {
    shared_ptr<ListNode> n1(new ListNode);
    shared_ptr<ListNode> n2(new ListNode);
    // 此时 n1引用计数1,n2引用计数1

    n1->_next = n2;  // n2引用计数变为2
    n2->_prev = n1;  // n1引用计数变为2
    // main离开后n1、n2析构,计数各减一,都变成1
    // 谁都没到零,ListNode永不被释放
}

两节点互相"拽"着对方,形成一个死锁。根源在于_next_prev参与了资源所有权的管理,但它们不应该拥有所属节点的生命周期决定权------它们只是引用而已。

2. weak_ptr:打破循环

weak_ptr专门用于绑定shared_ptr,但不增加引用计数 。它不支持直接管理资源(没有RAII),也没有operator*operator->。因为它不控制资源的生命周期,在资源已被释放后访问很危险,所以它提供了两个关键接口:

  • expired():检查引用的资源是否还存在。

  • lock():返回一个指向资源的shared_ptr。如果资源还在,返回的shared_ptr非空且安全访问;如果资源已被释放,返回空shared_ptr

cpp

复制代码
struct ListNode {
    int _data;
    weak_ptr<ListNode> _next;   // 改用weak_ptr
    weak_ptr<ListNode> _prev;
    ~ListNode() { cout << "~ListNode()" << endl; }
};

// n1->_next = n2; // 此时n2引用计数不增加,仍为1
// main结束后n1、n2正常析构

weak_ptr的构造只接受shared_ptr或另一个weak_ptr,不接受裸指针。这不是能力限制,是语义设计------你不拥有资源,也就不能凭空创建一个管理关系的入口。

需要访问时,先lock()

cpp

复制代码
weak_ptr<ListNode> wp = n1;
if (auto sp = wp.lock()) {
    sp->_data = 10;   // 安全访问
}

lock()返回的shared_ptr在作用域内持有引用计数,保证访问期间资源不会被释放,这是线程安全使用weak_ptr的基本模式。

3. 线程安全:引用计数要原子

shared_ptr的引用计数存放在堆上,多个shared_ptr对象在不同线程中进行拷贝和析构,会并发修改同一个计数,存在数据竞争。

最简单的修复是在模拟实现中将int* _pcount替换为atomic<int>* _pcount,核心操作变成原子增减:

cpp

复制代码
// 构造函数
_pcount = new atomic<int>(1);

// 拷贝构造
(*_pcount)++;

// release
if (--(*_pcount) == 0) { /* 释放 */ }

标准库的shared_ptr对引用计数的操作是线程安全的,但管理对象本身的操作不是 。如果多个线程同时修改同一个shared_ptr指向的对象内部数据,仍然需要外部同步。通俗来说:shared_ptr保证"自己不会因为并发拷贝/析构坏掉",但不保证"里面的对象线程安全"。后者是使用者的责任。

4. shared_ptr的另几个常用能力

shared_ptr支持operator bool,可以直接放在if里判断是否管理着资源:

cpp

复制代码
shared_ptr<int> sp;
if (!sp) cout << "empty" << endl;

make_shared<T>(args...)比直接new更好:一次内存分配同时容纳对象和控制块,效率高,也能避免newshared_ptr构造之间的异常导致泄漏。

自定义删除器让shared_ptr不仅能管内存,还能管文件、socket等任意资源:

cpp

复制代码
shared_ptr<FILE> sp(fopen("test.cpp", "r"),
                    [](FILE* f) { cout << "fclose" << endl; fclose(f); });

这本质上是RAII思想通过模板和函数对象实现的泛化------不再限于delete这一种释放方式。对于unique_ptr,同样支持删除器,只是语法上作为模板参数给出,略有不同。

5. 内存泄漏的最后一道防线

技术层面讲完了,补充一个工程层面的常识。

内存泄漏不是指物理内存消失了,而是程序分配了一段内存后失去了对它的追踪,这块内存既用不到也放不掉。短期运行的程序,进程退出时操作系统会回收所有内存,泄漏后果不大;但长期运行的进程(服务器、数据库、系统服务),泄漏会累积,最终内存耗尽。

避免泄漏的核心策略就两条:

  • 事前预防:用智能指针和RAII在设计层面消除手动释放的必要。

  • 事后检测:用valgrind、Visual Leak Detector等工具定期排查,尤其在版本上线前。

但检测工具只是安全网,不是救命稻草。设计阶段就把资源所有权理清楚,比任何工具都靠谱。

智能指针本身不是银弹。shared_ptr用得过滥,会导致对象生命周期模糊、循环引用、不必要的原子操作开销。一个简单的判断原则:默认用unique_ptr,只有当确实有多个所有者共享同一资源时才换shared_ptrweak_ptr永远作为辅助角色出现,解决特定的引用关系问题,不应独立承担资源管理职责。

相关推荐
菜菜的顾清寒5 小时前
力扣hot100(37)栈-有效的括号
java·开发语言
罗超驿5 小时前
9.LeetCode 209. 长度最小的子数组 | 滑动窗口专题详解
java·算法·leetcode·面试
叶落阁主6 小时前
AntV npm 投毒复盘:一次公司私服缓存恶意包引发的账号封禁事件
前端·安全·npm
水蓝烟雨6 小时前
0135. 分发糖果
算法·leetcode
IronMurphy6 小时前
【算法五十二】5. 最长回文子串
算法
guslegend6 小时前
第4讲:应用架构与代码组织
数据结构·人工智能·架构
Circ.6 小时前
Java 远程调用 NX 11 完整实战:参数读取、修改、STP 文件导出(附环境配置 + 源码)
java·开发语言·nx11
Lewiis6 小时前
白话选择排序
数据结构·算法·排序算法
2401_833269306 小时前
【无标题】
java·开发语言