【问题】C++:有哪些类型的智能指针,区别?

智能指针是一种在 C++ 中管理动态分配内存的工具,可以帮助避免内存泄漏和提高程序的安全性。在 C++11 标准引入之后,C++ 提供了三种主要类型的智能指针,它们分别是 std::unique_ptrstd::shared_ptrstd::weak_ptr。这些智能指针有不同的所有权管理方式和适用场景。

1. std::unique_ptr

  • 独占所有权std::unique_ptr 管理其所指向的对象,并且独占对该对象的访问权限。
  • 特点 :不能被复制,但可以通过移动语义转移所有权。当 std::unique_ptr 被销毁时,它所管理的对象也会被自动销毁。
  • 适用场景:适用于需要独占所有权且生命周期明确的场景,如避免内存泄漏的最佳选择。
cpp 复制代码
std::unique_ptr<int> ptr = std::make_unique<int>(10);

2. std::shared_ptr

  • 共享所有权std::shared_ptr 允许多个指针共享同一个对象。它维护一个引用计数,当最后一个 std::shared_ptr 被销毁时,其所管理的对象被释放。
  • 特点:可以被复制和赋值,引用计数递增和递减是自动管理的。
  • 适用场景:适用于多个指针需要访问同一对象的场景,如数据结构的节点。
cpp 复制代码
std::shared_ptr<int> ptr1 = std::make_shared<int>(20);
std::shared_ptr<int> ptr2 = ptr1; // 引用计数增加

3. std::weak_ptr

  • 弱引用std::weak_ptr 是为了解决 std::shared_ptr 的循环引用问题而引入的。它不增加引用计数,只是提供了对所指向对象的非拥有访问权限。
  • 特点 :不能直接访问对象,需要通过 lock() 方法转换为 std::shared_ptr 才能访问对象。
  • 适用场景 :主要用于避免 std::shared_ptr 循环引用导致的内存泄漏,例如图结构或者双向引用场景。
cpp 复制代码
std::shared_ptr<int> shared = std::make_shared<int>(30);
std::weak_ptr<int> weak = shared;
if (auto sharedPtr = weak.lock()) {
    // 使用 sharedPtr 访问对象
} else {
    // 对象已经被销毁
}

区别总结

  • 所有权管理std::unique_ptr 独占所有权,std::shared_ptr 共享所有权,std::weak_ptr 不影响所有权。
  • 复制和赋值std::unique_ptr 不能复制,只能移动;std::shared_ptr 可以复制和赋值;std::weak_ptr 不能直接访问对象,需要通过 lock() 方法转换为 std::shared_ptr 才能访问。
  • 线程安全std::shared_ptr 在多线程环境下使用时需要考虑引用计数的线程安全性,而 std::weak_ptr 可以避免循环引用导致的内存泄漏。

选择合适的智能指针取决于所需的所有权模型和对象生命周期管理策略。

相关推荐
naruto_lnq20 小时前
分布式系统安全通信
开发语言·c++·算法
学嵌入式的小杨同学20 小时前
【Linux 封神之路】信号编程全解析:从信号基础到 MP3 播放器实战(含核心 API 与避坑指南)
java·linux·c语言·开发语言·vscode·vim·ux
Re.不晚20 小时前
Java入门17——异常
java·开发语言
精彩极了吧21 小时前
C语言基本语法-自定义类型:结构体&联合体&枚举
c语言·开发语言·枚举·结构体·内存对齐·位段·联合
南极星10051 天前
蓝桥杯JAVA--启蒙之路(十)class版本 模块
java·开发语言
baidu_247438611 天前
Android ViewModel定时任务
android·开发语言·javascript
CSDN_RTKLIB1 天前
【四个场景测试】源文件编码UTF-8 BOM
c++
Dev7z1 天前
基于 MATLAB 的铣削切削力建模与仿真
开发语言·matlab
不能隔夜的咖喱1 天前
牛客网刷题(2)
java·开发语言·算法
小天源1 天前
Error 1053 Error 1067 服务“启动后立即停止” Java / Python 程序无法后台运行 windows nssm注册器下载与报错处理
开发语言·windows·python·nssm·error 1053·error 1067