答案:std::shared_ptr通过引用计数管理动态对象,推荐用std::make_shared创建,支持共享所有权,但需避免循环引用和裸指针重复初始化,配合weak_ptr可解决循环问题。

在C++中,std::shared_ptr 是一种智能指针,用于管理动态分配的对象,支持共享所有权。它通过引用计数机制自动跟踪有多少个 shared_ptr 指向同一个对象,当最后一个指向该对象的 shared_ptr 被销毁或重置时,对象会自动被删除,从而避免内存泄漏。
1. 基本用法:创建 shared_ptr
推荐使用 std::make_shared 来创建 shared_ptr,而不是直接使用构造函数。
-
推荐方式:
auto ptr = std::make_shared(args...); -
不推荐方式:
std::shared_ptrptr(new MyClass(args...));
使用 make_shared 更高效,因为它在一个内存分配中同时创建控制块和对象,并且更安全(避免异常安全问题)。
2. 共享所有权与引用计数
多个 shared_ptr 可以指向同一对象,每个拷贝都会使引用计数加1。
立即学习“C++免费学习笔记(深入)”;
std::shared_ptrp1 = std::make_shared (42); std::shared_ptr p2 = p1; // 引用计数变为2 std::cout << p1.use_count(); // 输出 2
调用 use_count() 可查看当前引用数量(仅用于调试,不要依赖它做逻辑判断)。
3. 避免循环引用
当两个对象通过 shared_ptr 相互持有对方时,引用计数永远不会归零,导致内存泄漏。
struct Node {
std::shared_ptr parent;
std::shared_ptr child;
};
// 如果 parent 和 child 互相指向,引用计数无法降为0
解决方法是使用 std::weak_ptr 打破循环。例如,将 parent 或 child 改为 weak_ptr,这样不会增加引用计数。
4. 注意事项与最佳实践
正确使用 shared_ptr 需要注意以下几点:
-
不要用裸指针初始化多个 shared_ptr:会导致重复释放。例如:
int* p = new int(10); shared_ptr是错误的。sp1(p); shared_ptr sp2(p); -
避免从 this 创建 shared_ptr:如果类需要返回自身的
shared_ptr,应继承std::enable_shared_from_this,然后使用shared_from_this()。 - 及时 reset() 不再需要的指针:有助于提前释放资源,尤其在长生命周期对象中。
- 避免频繁拷贝 shared_ptr:虽然拷贝是线程安全的,但会影响性能,特别是在高并发场景。
基本上就这些。合理使用 shared_ptr 能显著提升代码安全性,但也要注意其开销和陷阱,尤其是循环引用和异常安全问题。配合 unique_ptr 和 weak_ptr 使用,才能发挥智能指针的最大优势。不复杂但容易忽略细节。











