C++如何使用智能指针shared_ptr_C++共享所有权的内存管理

shared_ptr通过引用计数实现共享所有权,自动管理内存;推荐使用make_shared创建,避免循环引用时改用weak_ptr,并可自定义删除器处理特殊资源,提升安全性与性能。

在C++中,shared_ptr 是一种智能指针,用于实现共享所有权的动态内存管理。它通过引用计数机制自动追踪有多少个 shared_ptr 实例指向同一块堆内存,当最后一个指向该内存的 shared_ptr 被销毁或重置时,内存会自动释放,避免了内存泄漏。

创建 shared_ptr

可以使用 std::make_shared 或直接构造 shared_ptr 来创建对象:

  • std::make_shared(args) 是推荐方式,它更高效且异常安全。
  • 也可以用原始指针构造,但不建议直接传裸指针,容易出错。

示例:
auto ptr1 = std::make_shared(42);
std::shared_ptr ptr2(new int(10)); // 不推荐,但可用

共享所有权与引用计数

多个 shared_ptr 可以指向同一个对象,每增加一个副本,引用计数加1;销毁一个则减1。

  • 使用 use_count() 查看当前引用数量(调试用)。
  • 当引用计数变为0时,所管理的对象自动被析构。

示例:
auto a = std::make_shared<:string>("hello");
auto b = a; // 引用计数变为2
auto c = a; // 变为3
b.reset(); // b 置空,计数减为2

避免循环引用

当两个对象通过 shared_ptr 相互持有对方时,引用计数无法归零,导致内存泄漏。

  • 解决方法:在一方使用 std::weak_ptr,它不增加引用计数。
  • weak_ptr 可用于临时访问对象,检查是否还存在。

示例:
struct Node {
  std::shared_ptr parent;
  std::weak_ptr child; // 避免循环
};

自定义删除器

有时需要特殊方式释放资源(如关闭文件、释放数组等),可为 shared_ptr 指定删除器:

  • 删除器是一个函数或可调用对象,在引用计数为0时被调用。
  • 适用于数组、C风格资源(如 FILE*)等场景。

示例:
auto deleter = [](int* p) { delete[] p; };
std::shared_ptr arr(new int[10], deleter);

基本上就这些。合理使用 shared_ptr 能显著提升代码安全性,减少手动管理内存的错误。注意避免循环引用,并优先使用 make_shared 提高性能。