C++11如何使用std::unique_ptr和数组结合

来源:编程网作者:木下头衔:网络博主
导读:本期聚焦于小伙伴创作的《C++11如何使用std::unique_ptr和数组结合》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《C++11如何使用std::unique_ptr和数组结合》有用,将其分享出去将是对创作者最好的鼓励。

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

C++11如何使用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

免责声明:​ 已尽一切努力确保本网站所含信息的准确性。网站内容多为原创整理与精心编撰,观点力求客观中立。本站旨在免费分享,内容仅供个人学习、研究或参考使用。若引用了第三方作品,版权归原作者所有。如内容涉及您的权益,请联系我们处理。
内容垂直聚焦
专注技术核心技术栏目,确保每篇文章深度聚焦于实用技能。从代码技巧到架构设计,为用户提供无干扰的纯技术知识沉淀,精准满足专业提升需求。
知识结构清晰
覆盖从开发到部署的全链路。AI、前端、编程、数据库、服务器、建站、系统层层递进,构建清晰学习路径,帮助用户系统化掌握开发与运维所需的核心技术。
深度技术解析
拒绝泛泛而谈,深入技术细节与实践难点。无论是数据库优化还是服务器配置,均结合真实场景与代码示例进行剖析,致力于提供可直接应用于工作的解决方案。
专业领域覆盖
精准对应开发生命周期。从前端界面到后端编程,从数据库操作到服务器运维,形成完整闭环,一站式满足全栈工程师和运维人员的技术需求。
即学即用高效
内容强调实操性,步骤清晰、代码完整。用户可根据教程直接复现和应用于自身项目,显著缩短从学习到实践的距离,快速解决开发中的具体问题。
持续更新保障
专注既定技术方向进行长期、稳定的内容输出。确保各栏目技术文章持续更新迭代,紧跟主流技术发展趋势,为用户提供经久不衰的学习价值。