在C++11标准引入的智能指针体系中,std::unique_ptr是独占所有权的智能指针,能够自动释放所管理的动态资源。当需要处理动态分配的数组时,std::unique_ptr也提供了对应的支持,不过和单个对象的管理方式存在一定区别,需要开发者注意使用细节。

std::unique_ptr管理数组的基本声明方式
管理单个对象时,std::unique_ptr的声明不需要额外指定模板参数,默认会使用delete释放资源。但管理数组时,需要指定第二个模板参数为std::default_delete<T[]>,或者使用更简洁的数组形式声明,编译器会自动匹配对应的删除器,使用delete[]释放数组资源。
正确的数组声明示例如下:
#include <memory>
#include <iostream>
int main() {
// 正确声明管理int数组的unique_ptr
std::unique_ptr<int[]> arrPtr(new int[5]{1, 2, 3, 4, 5});
return 0;
}
如果错误地将数组声明为管理单个对象的形式,比如std::unique_ptr<int> arrPtr(new int[5]),那么在智能指针析构时会调用delete而非delete[],这会导致未定义行为,部分数组元素的内存无法正确释放,引发内存泄漏。
访问std::unique_ptr管理的数组元素
std::unique_ptr<T[]>重载了下标运算符operator[],可以直接通过下标访问数组元素,使用方式和普通数组指针一致。
示例代码如下:
#include <memory>
#include <iostream>
int main() {
std::unique_ptr<int[]> arrPtr(new int[5]{10, 20, 30, 40, 50});
// 通过下标访问元素
for (int i = 0; i < 5; ++i) {
std::cout << "arrPtr[" << i << "] = " << arrPtr[i] << std::endl;
}
// 修改数组元素
arrPtr[2] = 300;
std::cout << "修改后arrPtr[2] = " << arrPtr[2] << std::endl;
return 0;
}
需要注意的是,std::unique_ptr<T[]>不支持解引用运算符operator*和箭头运算符operator->,因为数组本身不是对象实例,没有成员可以访问。
自定义删除器的使用场景
默认情况下,std::unique_ptr<T[]>会使用delete[]释放数组,但如果数组是通过其他方式分配的,比如malloc、自定义内存池分配,就需要指定自定义删除器。
自定义删除器可以是函数、函数对象或者lambda表达式,需要匹配对应的释放逻辑。示例如下:
#include <memory>
#include <iostream>
#include <cstdlib>
// 自定义删除器,使用free释放内存
void freeArray(int* ptr) {
std::cout << "使用free释放数组内存" << std::endl;
free(ptr);
}
int main() {
// 使用malloc分配数组
int* rawArr = (int*)malloc(sizeof(int) * 3);
rawArr[0] = 1;
rawArr[1] = 2;
rawArr[2] = 3;
// 绑定自定义删除器
std::unique_ptr<int[], decltype(&freeArray)> arrPtr(rawArr, freeArray);
for (int i = 0; i < 3; ++i) {
std::cout << "arrPtr[" << i << "] = " << arrPtr[i] << std::endl;
}
// 析构时会调用freeArray释放内存
return 0;
}
和普通指针管理数组的对比
我们可以将普通指针管理数组和std::unique_ptr管理数组的差异整理成如下表格:
| 对比维度 | 普通指针管理数组 | std::unique_ptr管理数组 |
|---|---|---|
| 内存释放 | 需要手动调用delete[],容易遗漏 | 自动调用delete[],无需手动操作 |
| 所有权转移 | 需要手动复制指针,容易出现悬空指针 | 支持移动语义,所有权唯一,转移后原指针失效 |
| 异常安全 | 中途抛异常可能导致内存泄漏 | 析构时自动释放,异常场景下也安全 |
使用注意事项
- 不要将std::unique_ptr<T>用于管理数组,必须明确声明为std::unique_ptr<T[]>形式。
- std::unique_ptr<T[]>不支持使用get()方法获取指针后进行delete[]操作,否则会导致二次释放。
- 如果需要共享数组所有权,可以考虑使用std::shared_ptr,但需要手动指定delete[]删除器,因为std::shared_ptr默认使用delete释放资源。
通过正确使用std::unique_ptr管理数组,可以有效减少手动内存管理带来的错误,提升代码的健壮性和可维护性,这也是C++11之后推荐的内存管理方式之一。
C++11std::unique_ptr数组智能指针内存管理修改时间:2026-06-17 07:27:34