深入理解 C++ 智能指针:shared_ptr 与 unique_ptr 实践指南

133 阅读16分钟

一、引言

C++ 以其强大的性能而著称,但也因为其内存管理的复杂性而让许多开发者头疼。早期 C++ 依赖手动管理内存(new/delete),极易产生内存泄漏、悬挂指针等问题。为了提高资源管理的安全性和可维护性,C++11 引入了智能指针,提供自动的内存生命周期管理。

本篇将全面解析 C++ 智能指针的使用方式、原理、注意事项与实战应用,重点涵盖:

  • std::unique_ptr:唯一拥有者语义

  • std::shared_ptr:引用计数共享管理

  • std::weak_ptr:辅助管理循环引用

二、什么是智能指针?

智能指针是封装了原始指针的类对象,借助 RAII(资源获取即初始化)理念,在对象生命周期结束时自动释放资源,避免忘记手动释放。

标准库智能指针位于 <memory> 头文件下,主要包括:

智能指针

特点

unique_ptr

独占所有权,不可复制

shared_ptr

多个对象共享所有权,引用计数

weak_ptr

不拥有资源,解决循环引用问题

三、std::unique_ptr 使用详解

1. 基本用法

cpp复制编辑#include <memory>
#include <iostream>

class Dog {
public:
    Dog() { std::cout << "Dog created\n"; }
    ~Dog() { std::cout << "Dog destroyed\n"; }
    void bark() { std::cout << "Woof!\n"; }
};

int main() {
    std::unique_ptr<Dog> d1 = std::make_unique<Dog>();
    d1->bark();
}

输出:

nginx复制编辑Dog created  
Woof!  
Dog destroyed

2. 不可复制,只能移动

cpp复制编辑std::unique_ptr<Dog> d2 = d1;          // ❌ 编译错误
std::unique_ptr<Dog> d2 = std::move(d1); // ✅ 转移所有权

3. 释放和重置

cpp复制编辑d1.reset();             // 显式释放资源
d1 = std::make_unique<Dog>(); // 重建对象

4. 适合场景

  • 管理资源生命周期清晰的对象;

  • 避免复制(比如文件句柄、Socket);

  • 替代裸指针使用。

四、std::shared_ptr 使用详解

1. 基本用法

cpp复制编辑std::shared_ptr<Dog> d1 = std::make_shared<Dog>();
std::shared_ptr<Dog> d2 = d1; // 引用计数 +1

2. 引用计数机制

cpp复制编辑std::cout << d1.use_count(); // 输出 2

3. 自动销毁

use_count == 0,自动调用析构函数。

4. 自定义析构函数

cpp复制编辑std::shared_ptr<FILE> fp(
    fopen("test.txt", "w"),
    [](FILE* f) { if (f) fclose(f); }
);

五、std::weak_ptr:解决 shared_ptr 循环引用

1. 循环引用问题

cpp复制编辑struct A;
struct B;

struct A {
    std::shared_ptr<B> b_ptr;
};

struct B {
    std::shared_ptr<A> a_ptr;
};

// 构造时形成循环,导致无法释放

2. 使用 weak_ptr 打破环

cpp复制编辑struct B {
    std::weak_ptr<A> a_ptr; // 不增加引用计数
};

3. 使用 lock()

cpp复制编辑if (auto a = b->a_ptr.lock()) {
    a->do_something();
}

六、智能指针内存图示(图例)

cpp复制编辑std::shared_ptr<Dog> d1 = std::make_shared<Dog>();
std::shared_ptr<Dog> d2 = d1;


css复制编辑[d1] ---> [ControlBlock + Dog]
[d2] ---/
  • ControlBlock 中包含引用计数;

  • Dog 仅在计数为 0 时释放。

七、智能指针混用与陷阱

1. shared_ptr 和 unique_ptr 不可互转(需显式转换)

cpp复制编辑std::unique_ptr<Dog> up = std::make_unique<Dog>();
std::shared_ptr<Dog> sp = std::move(up); // ✅ 可行

反之不可直接转换。

2. 禁止多个 shared_ptr 管理同一裸指针

cpp复制编辑Dog* d = new Dog;
std::shared_ptr<Dog> sp1(d);
std::shared_ptr<Dog> sp2(d); // ❌ 崩溃!重复析构

推荐使用 std::make_shared

八、make_shared 与 make_unique 优势

  • 一次分配,提高性能;

  • 避免裸指针暴露;

  • 避免重构时构造多次的问题。

    cpp复制编辑auto sp = std::make_shared();

九、实际项目中的智能指针用法

1. 资源管理封装(Socket、文件、数据库连接)

cpp复制编辑class SocketHandle {
    std::unique_ptr<int, decltype(&close)> sock;
public:
    SocketHandle(int fd) : sock(new int(fd), &close) {}
};

2. UI 对象父子管理(Qt、游戏引擎)

cpp复制编辑class Widget {
    std::vector<std::shared_ptr<Widget>> children;
    std::weak_ptr<Widget> parent;
};

十、对比裸指针和智能指针

类型

自动释放

拷贝安全

支持引用计数

常用场景

原始指针

内部数据指针

unique_ptr

资源管理

shared_ptr

共享资源场景

weak_ptr

辅助防止环引用

十一、总结与推荐实践

✅ 推荐做法:

  • 使用 make_uniquemake_shared 替代 new

  • 优先选择 unique_ptr,只有在确实需要共享时使用 shared_ptr

  • 避免 shared_ptr 管理裸指针;

  • 使用 weak_ptr 解决循环引用。

🚫 不推荐做法:

  • 多个 shared_ptr 管理同一裸指针;

  • 不必要地共享资源,滥用 shared_ptr;

  • 把智能指针传值而非引用(增加不必要的引用计数开销)。

十二、结语

智能指针极大地提升了 C++ 代码的安全性与可维护性,是现代 C++ 编程不可或缺的一部分。理解并熟练掌握 unique_ptrshared_ptrweak_ptr 的使用场景与机制,是每位 C++ 程序员应具备的基本能力。

来源:素材入口页

来源:文档展示页

来源:渠道跳转页

来源:数据资源页

来源:段落展示页

来源:语段展示页

来源:跳转资料页

来源:入口信息页

来源:展示资源页

来源:资源路径

来源:片段展示页

来源:资料展示页

来源:内容说明页

来源:数据页链接

来源:链接资料页

来源:展示内容页

来源:通道入口页

来源:原文展示页

来源:数据链接页

来源:内容链接通道

来源:内容展示页

来源:跳转信息页

来源:说明内容页

来源:文章展示页

来源:素材链接页

来源:参考展示页

来源:信息跳转页

来源:渠道链接页

来源:详情展示页

来源:展示文档页

来源:信息数据页

来源:素材数据页

来源:片段链接页

来源:资料跳转通道

来源:展示文稿页

来源:信息渠道页

来源:渠道数据页

来源:条目展示页

来源:文库入口页

来源:数据说明页

来源:内容跳转页

来源:资料浏览

来源:内容索引

来源:路径页面

来源:资源概览

来源:参考素材

来源:文章链接

来源:原始文段

来源:文库索引

来源:资料路径页

来源:内容引用

来源:资源文段

来源:跳转资料

来源:链接入口页

来源:资料内容页

来源:文档访问

来源:资料详情页

来源:文库页面

来源:内容入口页

来源:段落资源

来源:条目链接

来源:内容详情页

来源:入口文库

来源:素材文段

来源:片段页面

来源:段落素材

来源:素材参考

来源:链接条目页

来源:文章内容页

来源:展示信息页

来源:参考段落页

来源:原始片段

来源:片段入口页

来源:文段链接页

来源:数据引用

来源:数据参考页

来源:原始内容页

来源:资源引用

来源:链接参考页

来源:资料引用

来源:入口索引页

来源:资源参考页

来源:展示入口页

来源:跳转链接页

来源:详情路径页

来源:路径通道页

来源:说明链接页

来源:索引文段

来源:入口文段

来源:内容引用页

来源:资源访问页

来源:访问详情页

来源:资料详情信息

来源:段落条目页

来源:说明通道页

来源:文章片段页

来源:展示索引页

来源:内容资源页

来源:资源页通道

来源:跳转索引页

来源:获取链接

来源:链接获取页

来源:内容详情信息

来源:原始数据页

来源:参考片段

来源:条目内容页

来源:段落跳转页

来源:段落链接通道

来源:条目引用

来源:内容提要页

来源:资源浏览页

来源:资料获取页

来源:片段索引页

来源:文库参考

来源:说明文章

来源:入口展示页

来源:信息获取页

来源:内容通道页

来源:说明参考页

来源:资料文稿页

来源:参考内容页

来源:通道展示页

来源:页面片段

来源:素材索引页

来源:通道资料页

来源:链接文章页

来源:展示说明页

来源:资料语段

来源:资源段落

来源:资料语料页

来源:信息详情页

来源:参考展示内容

来源:资料数据页

来源:通道跳转页

来源:原始数据内容

来源:说明资源页

来源:展示内容信息

来源:展示片段页

来源:资料通道链接

来源:访问通道页

来源:链接展示页

来源:说明资料页

来源:数据展示信息

来源:原文跳转页

来源:素材链接通道

来源:内容片段展示

来源:展示资料信息

来源:展示资源内容

来源:信息索引页

来源:素材参考页

来源:展示数据内容

来源:链接通道展示

来源:链接信息展示

来源:信息参考内容

来源:资源跳转信息

来源:条目跳转链接

来源:文档跳转信息

来源:片段跳转链接

来源:语段展示内容

来源:内容跳转展示

来源:页面跳转链接

来源:资料页面信息

来源:原文跳转链接

来源:条目跳转内容

来源:数据页展示链接

来源:链接内容展示页

来源:链接条目展示页

来源:信息跳转通道页

来源:资料跳转展示内容

来源:链接内容展示信息

来源:展示入口链接页面

来源:内容入口链接页面

来源:展示资料跳转链接

来源:入口展示跳转链接

来源:数据资料跳转链接

来源:入口内容跳转链接

来源:信息入口跳转链接

来源:路径入口跳转链接

来源:入口路径跳转链接

来源:通道入口跳转链接

来源:文稿内容跳转链接

来源:资料信息跳转链接

来源:数据内容跳转链接

来源:入口数据跳转链接

来源:文章信息跳转链接

来源:展示信息跳转链接

来源:展示资料入口链接

来源:内容链接入口展示页

来源:入口通道资源展示页

来源:参考来源入口展示链接

来源:内容资料跳转链接页面

来源:通道资源展示跳转链接

来源:链接文章入口资料页面

来源:访问通道信息展示链接

来源:文章展示页面跳转链接

来源:展示信息链接资源入口

来源:入口链接展示信息资源

来源:资料链接跳转展示通道

来源:文章入口跳转资料页面

来源:链接文章入口资源页面

来源:跳转文章入口链接资料

来源:资料跳转资源入口链接

来源:文档内容跳转链接入口

来源:资料链接展示入口页面

来源:数据入口展示资料链接

来源:展示资源链接跳转页面

来源:信息入口展示跳转链接

来源:资源跳转入口展示链接

来源:展示资源入口链接页面

来源:资料展示跳转链接入口

来源:资源展示跳转链接入口

来源:跳转链接展示资源页面

来源:链接资源展示页面入口

来源:展示入口链接跳转资源

来源:链接入口跳转资源展示

来源:页面入口链接跳转展示

来源:资源跳转链接入口页面

来源:内容跳转入口链接展示

来源:文章展示跳转资源入口

来源:链接展示入口跳转资源

来源:跳转资料展示入口链接

来源:跳转入口展示链接资源

来源:信息资源跳转入口链接

来源:资料展示入口链接页面

来源:跳转链接入口展示资料

来源:资料链接入口展示页面

来源:资源展示链接入口页面

来源:入口链接展示资料资源

来源:资源展示资料链接入口

来源:跳转链接资源入口展示

来源:链接展示资料资源入口

来源:入口展示资料链接资源

来源:相关阅读

来源:资料出处

来源:资料来源

来源:内容参考

来源:延伸阅读

来源:推荐链接

来源:相关信息

来源:参考内容

来源:推荐资料

来源:详细内容

来源:更多内容

来源:查看详情

来源:点击查看

来源:深入了解

来源:官方资料

来源:资料链接

来源:信息来源

来源:阅读更多

来源:文献出处

来源:内容链接

来源:数据出处

来源:技术参考

来源:案例来源

来源:实证依据

来源:报告链接

来源:研究链接

来源:统计数据

来源:数据来源

来源:分析资料

来源:理论出处

来源:论文参考

来源:说明详情

来源:调研内容

来源:支持数据

来源:背景文献

来源:学术出处

来源:报告来源

来源:文献引用

来源:权威来源

来源:实测数据

来源:延伸信息

来源:详情参考

来源:原文链接

来源:看原文

来源:阅读原文

来源:查看原帖

来源:原帖地址

来源:原始出处

来源:相关原文

来源:官方信息

来源:经验来源

来源:更多解析

来源:看更多

来源:网页资料

来源:实用信息

来源:入门参考

来源:指南文档

来源:推荐文章

来源:实例出处

来源:小贴士

来源:快速了解

来源:点此查看

来源:点击跳转

来源:原始链接

来源:网页参考

来源:本文依据

来源:内容跳转

来源:参考文档

来源:跳转详情

来源:网页原文

来源:链接详情

来源:点击去看

来源:点我查看

来源:来源说明

来源:资讯入口

来源:阅读跳转

来源:入门来源

来源:学习资料

来源:背景说明

来源:研究出处

来源:信息拓展

来源:衍生阅读

来源:延展资料

来源:链接来源

来源:原始内容

来源:探索原文

来源:获取信息

来源:查阅资料

来源:获取来源

来源:扩展阅读

来源:查阅详情

来源:来源详情

来源:页面跳转

来源:浏览原文

来源:详情链接

来源:说明出处

来源:内容补充

来源:参考原址

来源:附加链接

来源:来源页面

来源:点此了解更多

来源:原文阅读入口

来源:原始资料入口

来源:深度解析参考

来源:外部资源链接

来源:了解背景内容

来源:延伸阅读入口

来源:本文数据支持

来源:本文相关页面

来源:点这里查看

来源:快速跳转

来源:阅读此文

来源:查看推荐

来源:跳转页面

来源:继续阅读

来源:跳转参考文档

来源:外链参考

来源:去看看

来源:内容指引

来源:访问链接

来源:小贴士链接

来源:链接入口

来源:教程连接

来源:指向原文

来源:本文链接

来源:文献资料

来源:推荐参考

来源:同类资料

来源:引用内容

来源:来源参考

来源:实际出处

来源:文档入口

来源:学习入口

来源:阅读指引

来源:参考原文

来源:拓展来源

来源:额外参考

来源:内容引申

来源:来源位置

来源:附加内容

来源:网页跳转

来源:阅读地址

来源:档案出处

来源:数据支持

来源:引用来源

来源:上游资料

来源:查阅信息

来源:原页链接

来源:推荐说明

来源:提及内容

来源:本文参考

来源:链接参阅

来源:查阅出处

来源:内容输出

来源:内容路径

来源:深度阅读

来源:网文参考

来源:浏览入口

来源:原始输入

来源:本页推荐

来源:外链详情

来源:资讯路径

来源:源文链接

来源:关键参考

来源:相关文章

来源:指定链接

来源:点击链接

来源:原始地址

来源:学术链接

来源:网络出处

来源:案例出处

来源:支撑数据

来源:基础资料

来源:原始信息

来源:详细出处

来源:深入资料

来源:网页内容

来源:推荐原文

来源:网页参考资料

来源:来源跳转

来源:更多指引

来源:查阅页

来源:文本出处

来源:连接参考

来源:说明连接

来源:实用推荐

来源:点开阅读

来源:获取原始资料

来源:快速阅读链接

来源:内容引用来源

来源:实际案例参考

来源:对应来源

来源:跳转路径

来源:路径参考

来源:实际数据链接

来源:原页跳转

来源:外链地址

来源:跳转原始网页

来源:档案链接

来源:本站来源

来源:知识链接

来源:内容出自

来源:信息详情

来源:数据文献

来源:原本出处

来源:权威证据

来源:参考入口

来源:档案信息

来源:联网资料

来源:来源位置说明

来源:推荐文档

来源:详见资料

来源:继续查阅

来源:详情页

来源:资料页面

来源:原始说明

来源:原站内容

来源:内容追踪

来源:访问原页

来源:实例内容

来源:原文传送

来源:权威解读

来源:来自网页

来源:提及出处

来源:网页导航

来源:相关报道

来源:信息补充

来源:数据查看

来源:更多信息点

来源:资料一览

来源:内容载体

来源:报道来源

来源:说明原文

来源:知识参考

来源:网页地址

来源:点我阅读

来源:查阅指引

来源:点此前往

来源:看更多细节

来源:原始数据

来源:跳转资源

来源:文献详情

来源:外部详情

来源:更多原文

来源:入口链接

来源:数据详解

来源:文献链接

来源:学习资源

来源:内容导览

来源:查看原站

来源:链接出自

来源:进一步了解

来源:继续探索

来源:相关背景

来源:外部文档

来源:学习入口链接

来源:原始页面

来源:深层链接

来源:扩展查阅

来源:内部跳转

来源:链接点击

来源:查阅报告

来源:实时信息

来源:全文详情

来源:原始说明页

来源:来源入口

来源:内容浏览

来源:更多指向

来源:外链入口

来源:扫码进入

来源:专题链接

来源:实用出处

来源:历史内容

来源:扩展数据

来源:来源网页

来源:原页入口

来源:快捷入口

来源:相关跳转

来源:原数据页

来源:学术入口

来源:原内容页

来源:文档资源

来源:直接参考

来源:内容起点

来源:资源说明

来源:原链接地址

来源:通用资源

来源:链接访问

来源:推荐入口

来源:跳转原页

来源:引导信息

来源:快速查阅

来源:可用资料

来源:源地址

来源:网站跳转

来源:官方出品