首页 > 后端开发 > C++ > 正文

C++unique_ptr与数组结合使用方法

P粉602998670
发布: 2025-09-05 10:21:01
原创
988人浏览过
正确声明和初始化 unique_ptr 管理数组需使用 std::unique_ptr<T[]> 形式,并通过 new T[size] 初始化,例如 std::unique_ptr<int[]> arr(new int[10]);,这样析构时会自动调用 delete[] 释放内存,避免内存泄漏或崩溃。常见错误是使用 std::unique_ptr<T> 管理数组,导致 delete 与 delete[] 不匹配,引发未定义行为。unique_ptr 相比原始指针优势在于自动内存管理、异常安全、明确独占所有权,防止内存泄漏和悬挂指针。C++14 的 std::make_unique 不直接支持数组,但可自定义 make_unique_array 辅助函数实现类似功能。unique_ptr 管理数组时不可复制,所有权转移需使用 std::move,若需共享所有权应使用 std::shared_ptr,并配合自定义删除器或 std::default_delete<T[]> 确保正确调用 delete[]。

c++unique_ptr与数组结合使用方法

C++的

unique_ptr
登录后复制
可以管理动态分配的数组,但需要一些特殊的处理。关键在于使用
unique_ptr<T[]>
登录后复制
,而不是
unique_ptr<T>
登录后复制
。这告诉
unique_ptr
登录后复制
它管理的是一个数组,析构时会使用
delete[]
登录后复制
而不是
delete
登录后复制

解决方案

使用

std::unique_ptr<T[]>
登录后复制
来管理动态数组。

如何正确声明和初始化

unique_ptr
登录后复制
来管理数组?

立即学习C++免费学习笔记(深入)”;

声明时使用

unique_ptr<T[]>
登录后复制
,初始化时使用
new T[size]
登录后复制
。例如:

#include <iostream>
#include <memory>

int main() {
  // 创建一个可以管理 int 数组的 unique_ptr
  std::unique_ptr<int[]> arr(new int[10]);

  // 初始化数组
  for (int i = 0; i < 10; ++i) {
    arr[i] = i * 2;
  }

  // 访问数组元素
  for (int i = 0; i < 10; ++i) {
    std::cout << arr[i] << " ";
  }
  std::cout << std::endl;

  // unique_ptr 会自动释放数组内存
  return 0;
}
登录后复制

如何避免使用

unique_ptr
登录后复制
管理数组时的常见错误?

最常见的错误是使用

unique_ptr<T>
登录后复制
而不是
unique_ptr<T[]>
登录后复制
。 这样做会导致析构时调用
delete
登录后复制
而不是
delete[]
登录后复制
,从而导致未定义行为,通常是内存泄漏或程序崩溃。 另一个潜在的问题是忘记初始化数组。虽然
unique_ptr
登录后复制
会管理内存,但它不会自动初始化数组中的元素。

例如,以下代码是错误的:

#include <memory>

int main() {
  // 错误:使用了 unique_ptr<int> 来管理数组
  // 这会导致 delete 和 delete[] 不匹配
  // std::unique_ptr<int> arr(new int[10]); // 编译可以通过,但是运行时会出错

  // 正确的做法是
  std::unique_ptr<int[]> arr(new int[10]);

  return 0;
}
登录后复制

为什么

unique_ptr
登录后复制
是管理动态数组的好选择,相比原始指针有什么优势?

unique_ptr
登录后复制
提供了自动内存管理,避免了手动
delete[]
登录后复制
的需要。 这可以防止内存泄漏和悬挂指针。 此外,
unique_ptr
登录后复制
明确了所有权,使得代码更容易理解和维护。 原始指针需要手动管理内存,容易出错。

Zend Framework 2.4.3 完整版本
Zend Framework 2.4.3 完整版本

Zend框架2是一个开源框架,使用PHP 5.3 +开发web应用程序和服务。Zend框架2使用100%面向对象代码和利用大多数PHP 5.3的新特性,即名称空间、延迟静态绑定,lambda函数和闭包。 Zend框架2的组成结构是独一无二的;每个组件被设计与其他部件数的依赖关系。 ZF2遵循SOLID面向对象的设计原则。 这样的松耦合结构可以让开发人员使用他们想要的任何部件。我们称之为“松耦合”

Zend Framework 2.4.3 完整版本 344
查看详情 Zend Framework 2.4.3 完整版本

使用

unique_ptr
登录后复制
的优势:

  • 自动内存管理:
    unique_ptr
    登录后复制
    在离开作用域时会自动释放内存,无需手动
    delete[]
    登录后复制
  • 防止内存泄漏: 即使发生异常,
    unique_ptr
    登录后复制
    也能保证内存被释放。
  • 明确所有权:
    unique_ptr
    登录后复制
    明确表示对所管理资源的独占所有权。
  • 异常安全: 即使在构造或使用数组时抛出异常,
    unique_ptr
    登录后复制
    也能确保资源得到正确释放。

如何使用

std::make_unique
登录后复制
创建
unique_ptr
登录后复制
来管理数组?

C++14 引入了

std::make_unique
登录后复制
,但它不直接支持数组的创建。 这是因为
make_unique
登录后复制
的设计目的是为了避免
new
登录后复制
表达式和构造函数之间的异常安全问题,而对于数组,这个问题的处理方式略有不同。 你仍然需要使用
new T[]
登录后复制
来分配数组,然后用
unique_ptr
登录后复制
管理。

虽然不能直接使用

make_unique<T[]>(size)
登录后复制
,但可以自己实现一个类似的辅助函数:

#include <iostream>
#include <memory>

template <typename T>
std::unique_ptr<T[]> make_unique_array(size_t size) {
  return std::unique_ptr<T[]>(new T[size]);
}

int main() {
  // 使用辅助函数创建 unique_ptr<int[]>
  auto arr = make_unique_array<int>(5);

  for (int i = 0; i < 5; ++i) {
    arr[i] = i + 1;
    std::cout << arr[i] << " ";
  }
  std::cout << std::endl;

  return 0;
}
登录后复制

unique_ptr
登录后复制
管理数组时,如何传递和共享所有权?

unique_ptr
登录后复制
设计为独占所有权,这意味着只有一个
unique_ptr
登录后复制
可以指向特定的资源。 因此,不能直接复制
unique_ptr
登录后复制
。 如果需要转移所有权,可以使用
std::move
登录后复制

#include <iostream>
#include <memory>

int main() {
  std::unique_ptr<int[]> arr1(new int[5]);
  for (int i = 0; i < 5; ++i) {
    arr1[i] = i * 3;
  }

  // 转移所有权
  std::unique_ptr<int[]> arr2 = std::move(arr1);

  // 现在 arr1 不再拥有数组的所有权,访问 arr1 会导致未定义行为
  // arr1[0] = 10; // 避免这样做

  // arr2 现在拥有数组的所有权
  for (int i = 0; i < 5; ++i) {
    std::cout << arr2[i] << " ";
  }
  std::cout << std::endl;

  // arr2 会在离开作用域时释放数组内存

  return 0;
}
登录后复制

如果需要共享资源,考虑使用

std::shared_ptr
登录后复制
。 但是,对于数组,
shared_ptr
登录后复制
的使用需要特别小心,确保正确地指定删除器,以使用
delete[]
登录后复制
而不是
delete
登录后复制

#include <iostream>
#include <memory>

int main() {
  std::shared_ptr<int> arr(new int[5], [](int* p){ delete[] p; });

  for (int i = 0; i < 5; ++i) {
    arr.get()[i] = i * 5;
  }

  for (int i = 0; i < 5; ++i) {
    std::cout << arr.get()[i] << " ";
  }
  std::cout << std::endl;

  return 0;
}
登录后复制

或者使用别名模板,使代码更清晰:

#include <iostream>
#include <memory>

template<typename T>
using shared_array = std::shared_ptr<T>;

int main() {
    shared_array<int> arr(new int[5], std::default_delete<int[]>());

    for (int i = 0; i < 5; ++i) {
        arr.get()[i] = i * 5;
    }

    for (int i = 0; i < 5; ++i) {
        std::cout << arr.get()[i] << " ";
    }
    std::cout << std::endl;

    return 0;
}
登录后复制

总结,

unique_ptr
登录后复制
是管理动态数组的强大工具,但需要正确使用
unique_ptr<T[]>
登录后复制
并注意所有权转移。
shared_ptr
登录后复制
也可以用于共享数组,但需要小心处理删除器。

以上就是C++unique_ptr与数组结合使用方法的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号