unique_ptr和shared_ptr有何区别?

在C++中,unique_ptrshared_ptr是两种重要的智能指针,主要区别如下:


1. 所有权机制

  • unique_ptr(独占指针)

    采用独占所有权 模式,同一时刻只能有一个unique_ptr指向资源,不可复制(copy禁止),仅支持移动语义move)。
    示例

    cpp 复制代码
    std::unique_ptr<int> ptr1 = std::make_unique<int>(10);
    std::unique_ptr<int> ptr2 = std::move(ptr1); // ptr1所有权转移至ptr2
  • shared_ptr(共享指针)

    采用共享所有权 模式,多个shared_ptr可指向同一资源,通过引用计数 管理资源生命周期。
    示例

    cpp 复制代码
    std::shared_ptr<int> ptr1 = std::make_shared<int>(20);
    std::shared_ptr<int> ptr2 = ptr1; // 引用计数+1

2. 实现原理

  • unique_ptr

    无额外开销,仅封装原始指针,通过删除器 (默认为delete)释放资源。
    内存占用 :接近原始指针大小(通常为sizeof(void*))。

  • shared_ptr

    维护控制块 (含引用计数、弱计数、删除器等),资源释放需计数归零。
    内存占用:通常为两个指针大小(指向资源和控制块)。


3. 使用场景

  • unique_ptr适用场景

    • 资源独占管理(如工厂模式返回值)
    • 替代auto_ptr(已废弃)
    • 性能敏感场景(无计数开销)
  • shared_ptr适用场景

    • 跨作用域共享资源(如多线程数据传递)
    • 复杂所有权关系(如网状数据结构)
    • 需自动管理生命周期时

4. 性能对比

特性 unique_ptr shared_ptr
复制开销 禁止复制 原子操作更新计数
内存占用 低(1指针) 较高(2指针+控制块)
线程安全 非原子操作 引用计数原子操作

5. 代码示例对比

cpp 复制代码
// unique_ptr:所有权转移
std::unique_ptr<int> u1 = std::make_unique<int>(5);
std::unique_ptr<int> u2 = std::move(u1); // u1变为nullptr

// shared_ptr:共享所有权
std::shared_ptr<int> s1 = std::make_shared<int>(10);
{
    std::shared_ptr<int> s2 = s1; // 计数=2
} // s2析构,计数=1
点击并拖拽以移动

总结

  • 独占资源 时用unique_ptr(高效、轻量)
  • 共享资源 时用shared_ptr(灵活、安全)
  • 避免循环引用(shared_ptr可能需配合weak_ptr
相关推荐
悟空码字1 分钟前
别再System.out了!这份SpringBoot日志优雅指南,让你告别日志混乱
java·spring boot·后端
一 乐2 分钟前
工会管理|基于springboot + vue工会管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·工会管理系统
qq_380619164 分钟前
如何在phpMyAdmin中处理特殊字符账号名的授权_反引号的正确包裹
jvm·数据库·python
callJJ4 分钟前
Spring AI ETL 数据处理管道实战指南:从原始文档到向量索引
java·人工智能·spring·ai·etl·spring ai
暗暗别做白日梦5 分钟前
Maven 内部 Jar 包私服部署 + 多模块父工程核心配置
java·maven·jar
2201_756847337 分钟前
HTML函数在老旧浏览器运行慢是硬件问题吗_软硬协同分析【教程】
jvm·数据库·python
晓纪同学9 分钟前
EffctiveC++_第三章_资源管理
开发语言·c++·算法
从零开始的-CodeNinja之路15 分钟前
【Redis】Redis 缓存应用、淘汰机制—(四)
java·redis·缓存
蚊子码农16 分钟前
每日一题--C语言指针与内存泄漏:一道小问题的深度复盘
c语言·开发语言
Fanfanaas16 分钟前
Linux 系统编程 进程篇(一)
linux·运维·服务器·c语言·开发语言·网络·学习