答案:C++智能指针与容器结合使用可实现安全的内存管理。通过std::unique_ptr实现独占所有权,确保容器销毁时自动释放资源;用std::shared_ptr实现共享所有权,配合引用计数避免内存泄漏。两者均遵循RAII原则,提升异常安全性和代码清晰度。使用时需注意unique_ptr的移动语义、shared_ptr的循环引用及性能开销,推荐emplace_back和make_unique/make_shared优化构造。

C++智能指针与容器的结合使用,核心在于理解它们各自的生命周期管理机制,并选择合适的智能指针来适配容器的存储语义。说白了,就是如何让容器安全地管理那些我们通过
new
当我们需要在C++容器中存放动态分配的对象时,直接使用裸指针无疑是埋下了一颗定时炸弹。手动管理内存的复杂性,尤其是在异常发生时,极易导致内存泄漏。智能指针的引入,就是为了解决这个痛点,它利用RAII(Resource Acquisition Is Initialization)原则,将资源的生命周期与对象的生命周期绑定,从而实现自动化的内存管理。
具体到实践中,我们通常会根据对象的拥有权语义来选择
std::unique_ptr
std::shared_ptr
1. 独占所有权场景:使用 std::unique_ptr
立即学习“C++免费学习笔记(深入)”;
如果容器中的每个元素都应该独占它所指向的对象,也就是说,当元素从容器中移除或容器本身被销毁时,它所管理的对象也应该被销毁,那么
std::unique_ptr
#include <vector>
#include <memory>
#include <iostream>
class MyObject {
public:
int id;
MyObject(int i) : id(i) { std::cout << "MyObject " << id << " created." << std::endl; }
~MyObject() { std::cout << "MyObject " << id << " destroyed." << std::endl; }
void do_something() { std::cout << "MyObject " << id << " doing something." << std::endl; }
};
// 示例:std::vector 存储 std::unique_ptr
std::vector<std::unique_ptr<MyObject>> objects;
// 添加元素
// 方式一:直接创建并移动
objects.push_back(std::make_unique<MyObject>(1));
// 方式二:使用 emplace_back,可能更高效,避免临时对象
objects.emplace_back(std::make_unique<MyObject>(2));
// 方式三:从现有 unique_ptr 移动
auto temp_ptr = std::make_unique<MyObject>(3);
objects.push_back(std::move(temp_ptr)); // temp_ptr 此时为空
// 访问元素
objects[0]->do_something();
objects.back()->do_something();
// 移除元素(例如,移除最后一个)
// 当 unique_ptr 被销毁时,它指向的对象也会被销毁
objects.pop_back(); // MyObject 3 destroyed.
// 容器销毁时,所有 MyObject 都会被销毁
// (示例中省略了 main 函数的结束,但原理一致)这里要强调的是,
std::unique_ptr
unique_ptr
std::move
emplace_back
std::make_unique
2. 共享所有权场景:使用 std::shared_ptr
如果多个智能指针可以共同拥有一个对象,并且只有当所有拥有者都放弃所有权时,对象才会被销毁,那么
std::shared_ptr
#include <vector>
#include <memory>
#include <iostream>
class SharedResource {
public:
int value;
SharedResource(int v) : value(v) { std::cout << "SharedResource " << value << " created." << std::endl; }
~SharedResource() { std::cout << "SharedResource " << value << " destroyed." << std::endl; }
};
// 示例:std::vector 存储 std::shared_ptr
std::vector<std::shared_ptr<SharedResource>> resources;
// 创建一个共享资源
auto s_ptr1 = std::make_shared<SharedResource>(100);
// 添加到容器,s_ptr1 和 resources[0] 现在共享所有权
resources.push_back(s_ptr1);
// 再次添加,s_ptr1, resources[0], resources[1] 都共享所有权
resources.push_back(s_ptr1);
// 也可以直接在容器中创建
resources.emplace_back(std::make_shared<SharedResource>(200));
// 此时,s_ptr1 的引用计数是 3,resources[2] 的引用计数是 1
std::cout << "s_ptr1 ref count: " << s_ptr1.use_count() << std::endl; // 输出 3
// 访问元素
resources[0]->value = 101;
std::cout << "s_ptr1 value: " << s_ptr1->value << std::endl; // 输出 101
// 移除容器中的一个元素
resources.pop_back(); // SharedResource 200 destroyed. (因为它的引用计数降为 0)
std::cout << "s_ptr1 ref count after pop_back: " << s_ptr1.use_count() << std::endl; // 输出 2
// s_ptr1 超出作用域或被重置时,如果它是最后一个拥有者,SharedResource 100 才会销毁。
// (示例中省略了 main 函数的结束)std::shared_ptr
说实话,我刚开始学C++的时候,对智能指针这种“额外”的东西是有点抗拒的,觉得裸指针不是挺好用的吗?
new
delete
直接用裸指针存放对象到容器里,最直接的问题就是内存泄漏。想象一下,你
std::vector<MyObject*>
new MyObject()
clear()
delete
delete
智能指针,尤其是
std::unique_ptr
std::shared_ptr
std::unique_ptr<MyObject>
unique_ptr
delete
shared_ptr
delete
unique_ptr
unique_ptr
shared_ptr
shared_ptr
std::weak_ptr
std::vector<std::unique_ptr<MyObject>>
std::vector<std::shared_ptr<MyObject>>
所以,与其说智能指针是“额外”的,不如说它是现代C++中不可或缺的基石,它让我们的代码更健壮、更安全,也更易于理解。
std::unique_ptr
std::unique_ptr
首先,最核心的特性是不可复制性。
std::unique_ptr
unique_ptr
std::vector<std::unique_ptr<MyObject>> vec; auto p = std::make_unique<MyObject>(1); vec.push_back(p);
要解决这个问题,你需要显式地进行所有权转移。
std::move
unique_ptr
unique_ptr
std::move
vec.push_back(std::move(p));
std::move
unique_ptr
p
emplace_back
std::make_unique
vec.emplace_back(std::make_unique<MyObject>(4));
emplace_back
std::make_unique
unique_ptr
unique_ptr
另一个常见的误区是迭代器失效与所有权转移。当你从容器中删除一个
unique_ptr
std::vector<std::unique_ptr<T>>
i
auto p = vec[i];
auto p = std::move(vec[i]);
vec[i]
unique_ptr
std::move
vec.erase()
vec.remove()
再者,自定义删除器。
unique_ptr
std::unique_ptr<FILE, decltype(&fclose)> file_ptr(fopen("test.txt", "w"), &fclose);unique_ptr
最后,要小心容器内部的排序或重新分配操作。对于
std::vector
std::unique_ptr
std::list
std::map
unique_ptr
unique_ptr
std::shared_ptr
std::shared_ptr
shared_ptr
何时选择std::shared_ptr
shared_ptr
std::shared_ptr
shared_ptr
std::weak_ptr
shared_ptr
std::weak_ptr
std::shared_ptr
性能或设计考量:
虽然
std::shared_ptr
std::shared_ptr
shared_ptr
unique_ptr
unique_ptr
std::shared_ptr
shared_ptr
unique_ptr
shared_ptr
shared_ptr
shared_ptr
std::weak_ptr
std::weak_ptr
shared_ptr
std::shared_ptr
shared_ptr
std::shared_ptr
shared_ptr
std::make_shared
shared_ptr
std::make_shared
new
std::make_shared
std::vector<std::shared_ptr<MyObject>> vec; vec.emplace_back(std::make_shared<MyObject>(1));
在我看来,选择
shared_ptr
unique_ptr
shared_ptr
以上就是C++智能指针与容器结合使用方法的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号