c++++17 支持使用 std::shared_ptr

C++17 引入了对 std::shared_ptr 管理动态数组的正式支持,这使得我们可以在使用智能指针时更安全、更方便地管理数组资源。相比之前手动用 new[] 和 delete[] 的方式,现在结合 shared_ptr 可以避免内存泄漏和释放不当的问题。

下面我们就来看看怎么用 C++17 的 shared_ptr 来管理动态数组,以及一些需要注意的地方。

如何声明 shared_ptr 管理数组
在 C++17 之前,虽然可以用 shared_ptr 搭配自定义删除器来管理数组,但语法不够直观,也不够标准。从 C++17 开始,可以直接使用模板参数中的数组形式:
立即学习“C++免费学习笔记(深入)”;
std::shared_ptrarr(new int[10]);
这样就创建了一个指向 int 数组的 shared_ptr,并且它会自动调用 delete[] 而不是普通的 delete,避免了未定义行为。

你也可以写成带大小的形式(不过类型中不需要指定大小):
std::shared_ptrarr(new int[10]); // 类型里可以带大小,但不强制
但注意:不能像普通数组那样用下标访问元素,因为 operator[] 已被用于其他用途。要访问数组元素,应该使用 get() 获取原始指针后再操作:
arr.get()[0] = 42;
使用建议与注意事项
-
避免混用 delete 和 delete[]:这是最基本的安全原则。
shared_ptr内部默认使用delete[],所以你不用再自己加删除器。 - *不要把 shared_ptr
当作 T 使用**:比如传给需要裸指针的函数时,要用.get()显式获取。 -
数组大小不会自动记录:不像
std::vector,shared_ptr不保存数组长度信息,你需要自己维护或封装。 - 不能直接赋值整个数组内容:只能一个一个元素赋值,或者配合算法库处理。
举个例子,初始化并填充数组:
网趣购物系统静态版支持网站一键静态生成,采用动态进度条模式生成静态,生成过程更加清晰明确,商品管理上增加淘宝数据包导入功能,与淘宝数据同步更新!采用领先的AJAX+XML相融技术,速度更快更高效!系统进行了大量的实用性更新,如优化核心算法、增加商品图片批量上传、谷歌地图浏览插入等,静态版独特的生成算法技术使静态生成过程可随意掌控,从而可以大大减轻服务器的负担,结合多种强大的SEO优化方式于一体,使
auto arr = std::make_shared(5); for (int i = 0; i < 5; ++i) { arr.get()[i] = i * 2; }
更推荐的方式:结合 make_shared 创建数组
除了直接用 new,你还可以使用 std::make_shared 来构造数组,这种方式更简洁也更安全:
auto arr = std::make_shared(20); // 创建一个含20个int的数组
这种方式的好处是:
- 分配内存和初始化在一个原子操作中完成,减少异常风险;
- 避免裸
new,提高代码可读性和安全性; - 更容易写出异常安全的代码。
但要注意的是:目前 make_shared 不支持带初始化列表的方式构造数组(如 make_shared 是错误的),所以如果你需要初始化特定值,得自己手动设置。
实际应用场景举例
常见于需要共享资源所有权的场景,比如:
- 多个对象需要共享一块缓冲区;
- 作为函数返回值传递数组资源;
- 在异步任务中传递数据块,避免生命周期问题;
例如:
void process_data(std::shared_ptrdata, int size) { for (int i = 0; i < size; ++i) { std::cout << data.get()[i] << " "; } } int main() { auto buffer = std::make_shared (5); for (int i = 0; i < 5; ++i) buffer.get()[i] = i + 1; process_data(buffer, 5); }
在这个例子中,buffer 可以安全地在多个地方使用,不用担心提前释放的问题。
基本上就这些。C++17 的 shared_ptr 对数组的支持让资源管理更清晰,用起来也不复杂,但在实际使用中还是要注意数组大小管理和访问方式。









