【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永远作为辅助角色出现,解决特定的引用关系问题,不应独立承担资源管理职责。

相关推荐
JieE2128 小时前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack2015 小时前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树17 小时前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
Aphasia31121 小时前
VPN 与内网穿透
安全
JieE2121 天前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2121 天前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术2 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦2 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
clint4562 天前
C++进阶(1)——前景提要
c++