C++如何在智能指针中管理动态数组

最推荐使用 std::unique_ptr 管理动态数组,因其能自动调用 delete[] 避免内存泄漏;若需共享所有权,可用带自定义删除器的 std::shared_ptr;但多数情况下应优先选用 std::vector,因其兼具自动管理、丰富接口与优良性能。

c++如何在智能指针中管理动态数组

在C++中,管理动态数组与智能指针结合使用,最直接且推荐的方式是利用

std::unique_ptr

。它专为动态数组设计,能确保在对象生命周期结束时自动调用正确的

delete[]

操作,从而避免内存泄漏。如果确实需要共享所有权,

std::shared_ptr

也能做到,但它需要一个自定义的删除器来正确处理数组释放。不过,话说回来,在大多数现代C++场景里,

std::vector

往往是更安全、更方便且性能同样出色的默认选择。

解决方案

当我们需要在堆上分配一个动态数组时,传统的做法是使用

new T[size]

,然后手动

delete[]

。这极易出错,稍不留神就会忘记释放内存,或者更糟的是,用

delete

而非

delete[]

释放数组,导致未定义行为。智能指针的出现就是为了解决这类问题。

1.

std::unique_ptr

:独占所有权的最佳选择

这是管理动态数组最直接、最安全的方法,尤其当你确定数组的生命周期与智能指针的生命周期完全绑定,且没有其他地方需要共享该数组时。

std::unique_ptr

有一个特化版本

std::unique_ptr

,它明确知道自己管理的是一个数组,因此在析构时会自动调用

delete[]

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

#include #include void process_data(int* arr, size_t size) {    for (size_t i = 0; i < size; ++i) {        arr[i] *= 2;    }}int main() {    // 创建一个包含10个int的动态数组    std::unique_ptr arr_ptr = std::make_unique(10); // C++14及更高版本推荐    // 或者 C++11 风格:    // std::unique_ptr arr_ptr(new int[10]);     for (int i = 0; i < 10; ++i) {        arr_ptr[i] = i + 1; // 像普通数组一样访问元素    }    std::cout << "Original array elements: ";    for (int i = 0; i < 10; ++i) {        std::cout << arr_ptr[i] << " ";    }    std::cout << std::endl;    // 可以获取原始指针传递给C风格API    process_data(arr_ptr.get(), 10);    std::cout << "Processed array elements: ";    for (int i = 0; i < 10; ++i) {        std::cout << arr_ptr[i] << " ";    }    std::cout << std::endl;    // arr_ptr超出作用域时,会自动调用 delete[] arr_ptr.get()    return 0;}
std::make_unique(10)

是C++14引入的,它避免了显式

new

,并提供了异常安全保证,我个人觉得,这是更现代、更安全的写法。

2.

std::shared_ptr

与自定义删除器:共享所有权

如果你的动态数组需要被多个

std::shared_ptr

实例共享,那么情况就稍微复杂一点。

std::shared_ptr

的默认删除器只会调用

delete

,而不是

delete[]

。这意味着,如果你直接这样用:

std::shared_ptr shared_arr(new int[10]);

,那么在

shared_arr

析构时,会调用

delete

而不是

delete[]

,这会导致未定义行为。

正确的做法是提供一个自定义的删除器(通常是一个lambda表达式),明确告诉

std::shared_ptr

如何释放数组内存:

#include #include #include  // 后面会提到int main() {    // 使用自定义删除器管理动态数组    std::shared_ptr shared_arr(new int[10], [](int* p) {        std::cout << "Custom deleter called for shared_arr, deleting array." << std::endl;        delete[] p;    });    for (int i = 0; i < 10; ++i) {        shared_arr.get()[i] = (i + 1) * 10; // 通过get()获取原始指针访问    }    // 可以创建其他shared_ptr实例共享所有权    std::shared_ptr another_shared_arr = shared_arr;    std::cout << "Shared array elements: ";    for (int i = 0; i < 10; ++i) {        std::cout << shared_arr.get()[i] << " ";    }    std::cout << std::endl;    // 当所有shared_ptr实例都超出作用域时,自定义删除器会被调用一次    return 0;}

这里有个细节,

std::shared_ptr

而不是

std::shared_ptr

。这是因为

std::shared_ptr

没有像

std::unique_ptr

那样为数组提供特化版本。因此,当你使用

std::shared_ptr

管理数组时,你实际上是管理一个指向数组第一个元素的指针,并依赖自定义删除器来正确地释放整个数组。访问元素时,你需要通过

get()

方法获取原始指针,然后进行指针算术或使用下标操作符。

3.

std::vector

:大多数情况下的首选

我个人认为,除非有非常特殊的原因,比如需要与C风格API高度兼容,或者对内存布局有极致的控制需求,否则

std::vector

几乎总是管理动态数组的最佳选择。它提供了RAII(资源获取即初始化),自动内存管理,以及丰富的API(如迭代器、容量管理、元素访问方法等),并且通常在性能上与原始数组不相上下。

#include #include int main() {    std::vector vec(10); // 创建一个包含10个int的动态数组    for (int i = 0; i < 10; ++i) {        vec[i] = i * 100; // 像普通数组一样访问元素    }    std::cout << "Vector elements: ";    for (int i = 0; i < vec.size(); ++i) {        std::cout << vec[i] << " ";    }    std::cout << std::endl;    // vec超出作用域时,会自动释放内存    return 0;}
std::vector

内部已经处理了内存的分配和释放,并且提供了类型安全和边界检查(在调试模式下)。它的接口设计也更加现代化和易用。

为什么不能直接用

std::unique_ptr

管理数组?

这是一个非常常见的误区,我见过不少新手会犯这样的错误。核心问题在于

std::unique_ptr

std::unique_ptr

在析构时调用的内存释放函数不同。

当你声明

std::unique_ptr ptr(new T[size]);

时,你创建了一个

unique_ptr

,它被设计来管理单个对象

T

。因此,当

ptr

超出作用域时,它的析构函数会调用

delete ptr.get();

然而,如果你用

new T[size]

分配了一个数组,正确的释放方式是

delete[] ptr.get();

所以,当

delete

被用于释放通过

new[]

分配的内存时,就会导致未定义行为(Undefined Behavior, UB)。这意味着程序可能会崩溃,可能会泄漏内存,也可能表面上看起来正常运行,但在未来的某个时刻,在某个不相关的代码路径中,问题会突然暴露出来,这种错误调试起来非常痛苦。

简单来说,

delete

delete[]

不是可以互换的。

delete

针对单个对象,

delete[]

针对数组。它们在底层可能执行完全不同的操作,例如

new[]

可能会在实际数据之前存储数组的大小信息,而

delete[]

会利用这些信息。如果只调用

delete

,这些信息可能不会被正确处理,导致内存损坏。

// 错误的示例,会导致未定义行为!std::unique_ptr bad_array_ptr(new int[10]); // 当 bad_array_ptr 析构时,会调用 delete (int*) 而不是 delete[] (int*)// 这就是问题所在。

所以,请务必记住:管理动态数组,要用

std::unique_ptr

,而不是

std::unique_ptr

。这是C++类型系统和内存管理规则的一个重要细节。

std::shared_ptr

中管理动态数组需要注意什么?

正如前面提到的,

std::shared_ptr

在设计上没有为数组提供像

std::unique_ptr

那样的特化版本。这意味着,如果你想用

std::shared_ptr

来管理

new T[size]

分配的动态数组,你必须提供一个自定义的删除器。

最关键的注意事项就是:不要忘记自定义删除器,并且确保删除器调用的是

delete[]

// 错误示范:没有自定义删除器// std::shared_ptr my_shared_array(new int[5]); // 同样是未定义行为!// 正确示范:使用lambda表达式作为自定义删除器std::shared_ptr my_shared_array_correct(new int[5], [](int* p) {    std::cout << "Custom deleter for shared_ptr array called." << std::endl;    delete[] p; // 确保是 delete[]});

自定义删除器是一个可调用对象(函数指针、函数对象或lambda),它接受一个原始指针作为参数,并负责释放该指针指向的资源。

std::shared_ptr

会在最后一个引用计数归零时调用这个删除器。

如果你忘记提供自定义删除器,

std::shared_ptr

会使用其默认的删除器,它会调用

delete

。这和

std::unique_ptr

犯的错误一样,导致未定义行为。

此外,需要注意的是,当使用

std::shared_ptr

管理数组时,你通常需要通过

get()

方法获取原始指针来访问数组元素,例如

my_shared_array_correct.get()[i]

。这是因为

std::shared_ptr

operator[]

没有为数组语义重载,它只是一个指向单个

T

的智能指针。

虽然

std::shared_ptr

提供了这种灵活性,但它也带来了一定的开销。

std::shared_ptr

需要维护一个控制块来存储引用计数和删除器等信息,这会占用额外的内存。而且,引用计数的增减通常涉及到原子操作,这在多线程环境下会引入同步开销。因此,除非你确实需要共享动态数组的所有权,否则

std::unique_ptr

std::vector

往往是更轻量级的选择。

什么时候应该优先考虑

std::vector

而不是智能指针管理动态数组?

在我看来,这是一个非常实际的问题,也是现代C++编程中一个重要的设计决策。我几乎可以说,在绝大多数情况下,

std::vector

都应该成为你管理动态数组的首选,而不是直接使用智能指针来包装

new T[]

以下是我个人总结的一些理由,说明为什么

std::vector

常常是更好的选择:

RAII 的完整实现与自动内存管理:

std::vector

内部已经完美地实现了RAII。你不需要担心

new

delete[]

的配对,它会自动处理内存的分配和释放。这大大减少了内存泄漏和悬空指针的风险。丰富的API和功能:

std::vector

不仅仅是一个内存容器,它是一个功能完备的序列容器。它提供了:动态大小调整: 你可以方便地添加或删除元素,

std::vector

会自动处理内存的重新分配。这是

new T[]

无法直接提供的。迭代器支持: 可以轻松地与STL算法(如

std::sort

,

std::for_each

等)配合使用。边界检查:

at()

方法提供运行时边界检查(尽管

operator[]

不提供,但调试时可以帮助发现问题)。容量管理:

capacity()

,

reserve()

,

shrink_to_fit()

等方法可以让你更好地控制内存使用。构造函数和赋值操作: 提供了方便的复制、移动语义。类型安全:

std::vector

明确表示它包含

T

类型的元素,并提供类型安全的访问。性能通常足够好: 很多人可能误以为

std::vector

会有很大的性能开销。但实际上,现代C++编译器对

std::vector

的优化非常到位。它的元素是连续存储的,这与原始数组一样,因此缓存局部性很好。对于绝大多数应用来说,

std::vector

的性能表现与原始数组几乎没有区别,甚至在某些情况下可能因为更好的内存管理策略而表现更优。与现代C++生态系统的无缝集成:

std::vector

是STL的核心组件,与C++标准库中的其他部分(如算法、迭代器)配合得天衣无缝。

那么,什么时候会考虑智能指针管理

new T[]

呢?

C风格API的互操作性: 当你需要将一个动态数组的原始指针传递给一个只接受

T*

void*

的C风格函数或库时,

std::unique_ptr::get()

std::shared_ptr::get()

就可以派上用场。虽然

std::vector::data()

也能提供这个功能,但在某些特定场景下,智能指针可能更直接。极度内存敏感或特殊分配需求: 某些非常底层或嵌入式系统编程中,你可能需要使用自定义的内存分配器(例如,从预分配的内存池中分配),而这些分配器可能不兼容

std::vector

的默认行为。在这种情况下,手动

new T[size]

并用智能指针管理可能是一种选择。但这非常罕见,且需要深入理解内存管理。遗留代码的现代化: 如果你正在处理大量使用

new T[]

的遗留C++代码,并且希望逐步引入RAII,那么将这些裸指针包装到

std::unique_ptr

std::shared_ptr

(带自定义删除器)中,是一个相对低风险的重构策略。

总而言之,我的建议是:总是从

std::vector

开始考虑。只有当

std::vector

明确无法满足你的特定需求(通常是与C接口的深度集成或极端的自定义内存管理)时,才转向

std::unique_ptr

。而

std::shared_ptr

管理动态数组,则是在需要共享所有权且必须使用

new T[]

时的最后手段。 这样做能让你的代码更健壮、更易读、更易维护。

以上就是C++如何在智能指针中管理动态数组的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1474281.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月18日 21:20:18
下一篇 2025年12月18日 21:20:27

相关推荐

  • CSS mask属性无法获取图片:为什么我的图片不见了?

    CSS mask属性无法获取图片 在使用CSS mask属性时,可能会遇到无法获取指定照片的情况。这个问题通常表现为: 网络面板中没有请求图片:尽管CSS代码中指定了图片地址,但网络面板中却找不到图片的请求记录。 问题原因: 此问题的可能原因是浏览器的兼容性问题。某些较旧版本的浏览器可能不支持CSS…

    2025年12月24日
    900
  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 为什么设置 `overflow: hidden` 会导致 `inline-block` 元素错位?

    overflow 导致 inline-block 元素错位解析 当多个 inline-block 元素并列排列时,可能会出现错位显示的问题。这通常是由于其中一个元素设置了 overflow 属性引起的。 问题现象 在不设置 overflow 属性时,元素按预期显示在同一水平线上: 不设置 overf…

    2025年12月24日 好文分享
    400
  • 网页使用本地字体:为什么 CSS 代码中明明指定了“荆南麦圆体”,页面却仍然显示“微软雅黑”?

    网页中使用本地字体 本文将解答如何将本地安装字体应用到网页中,避免使用 src 属性直接引入字体文件。 问题: 想要在网页上使用已安装的“荆南麦圆体”字体,但 css 代码中将其置于第一位的“font-family”属性,页面仍显示“微软雅黑”字体。 立即学习“前端免费学习笔记(深入)”; 答案: …

    2025年12月24日
    000
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 为什么我的特定 DIV 在 Edge 浏览器中无法显示?

    特定 DIV 无法显示:用户代理样式表的困扰 当你在 Edge 浏览器中打开项目中的某个 div 时,却发现它无法正常显示,仔细检查样式后,发现是由用户代理样式表中的 display none 引起的。但你疑问的是,为什么会出现这样的样式表,而且只针对特定的 div? 背后的原因 用户代理样式表是由…

    2025年12月24日
    200
  • inline-block元素错位了,是为什么?

    inline-block元素错位背后的原因 inline-block元素是一种特殊类型的块级元素,它可以与其他元素行内排列。但是,在某些情况下,inline-block元素可能会出现错位显示的问题。 错位的原因 当inline-block元素设置了overflow:hidden属性时,它会影响元素的…

    2025年12月24日
    000
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 为什么使用 inline-block 元素时会错位?

    inline-block 元素错位成因剖析 在使用 inline-block 元素时,可能会遇到它们错位显示的问题。如代码 demo 所示,当设置了 overflow 属性时,a 标签就会错位下沉,而未设置时却不会。 问题根源: overflow:hidden 属性影响了 inline-block …

    2025年12月24日
    000
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 为什么我的 CSS 元素放大效果无法正常生效?

    css 设置元素放大效果的疑问解答 原提问者在尝试给元素添加 10em 字体大小和过渡效果后,未能在进入页面时看到放大效果。探究发现,原提问者将 CSS 代码直接写在页面中,导致放大效果无法触发。 解决办法如下: 将 CSS 样式写在一个单独的文件中,并使用 标签引入该样式文件。这个操作与原提问者观…

    2025年12月24日
    000
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 为什么我的 em 和 transition 设置后元素没有放大?

    元素设置 em 和 transition 后不放大 一个 youtube 视频中展示了设置 em 和 transition 的元素在页面加载后会放大,但同样的代码在提问者电脑上没有达到预期效果。 可能原因: 问题在于 css 代码的位置。在视频中,css 被放置在单独的文件中并通过 link 标签引…

    2025年12月24日
    100
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信