std::unique_ptr
和std::shared_ptr
是C++11中引入的两种智能指针类型,它们都可以自动管理内存,以避免内存泄漏。然而,它们在所有权语义和使用场景上有明显的区别:
std::unique_ptr
- 独占所有权:
std::unique_ptr
代表对其所指向对象的唯一所有权。一次只能有一个std::unique_ptr
指向一个给定的对象。 - 轻量级:因为它不需要维护引用计数,所以它比
std::shared_ptr
更轻量级。 - 不可复制:
std::unique_ptr
不能被复制,但可以被移动,这意味着所有权可以被转移。 - 自动销毁:当
std::unique_ptr
离开作用域时,它所指向的对象会被自动销毁(调用delete
)。
std::shared_ptr
- 共享所有权:
std::shared_ptr
允许多个指针实例共享对同一个对象的所有权。这通过引用计数来实现,每当一个新的std::shared_ptr
被创建或者复制时,引用计数会增加;当std::shared_ptr
被销毁时,引用计数会减少。 - 线程安全:
std::shared_ptr
在修改内部引用计数时是线程安全的,但是通过std::shared_ptr
访问对象的成员不是线程安全的,除非另外加锁。 - 性能开销:由于维护引用计数,
std::shared_ptr
比std::unique_ptr
有更大的性能开销。 - 循环引用问题:如果两个或多个
std::shared_ptr
相互引用,形成一个环,它们可能会导致内存泄漏,因为引用计数永远不会降到零。这可以通过使用std::weak_ptr
来解决。
使用场景
- 当你需要一个对象有一个清晰的单一所有者时,应该使用
std::unique_ptr
。这通常是最优的选择,因为它的开销最小。 - 当多个所有者需要共享同一个对象时,应该使用
std::shared_ptr
。这适用于对象的生命周期需要由多个持有者共同管理的情况。
总之,std::unique_ptr
和std::shared_ptr
都是用于资源管理的工具,但是它们的设计目的和适用场景不同。在选择使用哪一个时,应该考虑所有权的需求和性能的影响。
#include <iostream>
#include <memory>
class MyClass {
public:
void showMessage() {
std::cout << "Hello from MyClass!" << std::endl;
}
};
int main() {
// 使用 std::unique_ptr 创建一个独占所有权的智能指针
std::unique_ptr<MyClass> uniquePtr = std::make_unique<MyClass>();
// 使用 std::shared_ptr 创建一个共享所有权的智能指针
std::shared_ptr<MyClass> sharedPtr1 = std::make_shared<MyClass>();
std::shared_ptr<MyClass> sharedPtr2 = sharedPtr1; // 共享所有权
// 使用智能指针调用对象的方法
uniquePtr->showMessage();
sharedPtr1->showMessage();
sharedPtr2->showMessage();
// 当指针超出作用域时,对象会自动释放
return 0;
}