0

0

C++中的std::make_shared为什么比直接new更好?(减少内存分配次数)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-13 14:01:02

|

266人浏览过

|

来源于php中文网

原创

c++kquote>std::make_shared 能减少内存分配次数,因为它将控制块和对象数据合并到一次堆分配中,而直接 new 再构造 shared_ptr 会触发两次独立分配。

c++中的std::make_shared为什么比直接new更好?(减少内存分配次数)

std::make_shared 为什么能减少内存分配次数

因为 std::make_shared 把控制块(control block)和对象数据分配在同一块连续内存里,而直接用 new 构造再传给 std::shared_ptr 构造函数时,控制块和对象是两次独立的堆分配。

控制块里存引用计数、弱引用计数、删除器等元数据;对象本身是用户数据。两次分配不仅慢,还增加缓存不友好性和碎片化风险。

  • 两次分配:先 new T 分配对象,再由 shared_ptr 内部另一次 operator new 分配控制块
  • 一次分配:make_shared 预计算总大小(控制块 + 对齐填充 + T),单次申请,然后在其中分别构造控制块和对象
  • 注意:仅对无自定义删除器、无对齐要求超出默认的情况才保证合并分配;若用了 std::allocate_shared 或自定义分配器,行为取决于分配器实现

直接 new + shared_ptr 构造的典型写法与开销

这种写法看似直观,但隐含两次堆分配:

std::shared_ptr ptr(new std::string("hello")); // ❌ 两次分配

即使编译器做了某些优化(如 NRVO 或分配器内联),标准不保证合并;且无法避免控制块中存储的“指向对象的指针”带来的间接访问。

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

MuleRun
MuleRun

全球首个AI Agent交易平台

下载
  • 控制块中必须保存一个指向对象的指针(用于析构时调用 T 的析构函数)
  • 两次分配导致两块不相邻内存,降低 CPU 缓存命中率
  • 异常安全虽有保障(shared_ptr 构造失败会自动清理已分配对象),但性能损失固定存在

make_shared 的限制与容易踩的坑

make_shared 不是万能替代方案,以下情况它无法工作或会出问题:

  • T 的构造函数是私有的,且 make_shared 无法访问(它不参与友元声明,也不像 shared_ptr 构造函数那样可被显式授权)
  • 需要自定义删除器(如文件句柄、GPU 内存释放)——make_shared 固定使用 delete,必须用 shared_ptr(new T, my_deleter)
  • 类重载了 operator new 且逻辑依赖于单独分配对象(例如按类型分页管理),make_shared 绕过该重载,改用全局或类模板内的分配逻辑
  • 对象大小极大(如百 MB 数组),而控制块很小,合并分配可能导致大块内存无法复用(尤其在小对象频繁分配场景下)

实测分配次数差异(以 libc++ 和 libstdc++ 为例)

可通过重载全局 operator new 或使用 malloc_hook(glibc)粗略验证。更可靠的是查看生成的汇编或用 valgrind --tool=massif 观察堆快照:

auto a = std::make_shared(42);        // 1 次 malloc(约 32–48 字节,含控制块)  
auto b = std::shared_ptr(new int(42)); // 2 次 malloc(1 次给 int,1 次给控制块)

实际大小取决于标准库实现:libstdc++ 控制块通常 16 字节(不含虚表指针),libc++ 是 24 字节;加上对齐填充和 int 本身,make_shared 一般只触发一次 32 或 48 字节分配。

真正复杂的地方在于:当 T 是带非平凡构造函数的大结构体,且你又需要自定义销毁逻辑时,就不得不放弃 make_shared ——这时候减少分配次数的目标就得让位于资源管理正确性。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

194

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

187

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

534

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

194

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

386

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

569

2023.08.10

交互式图表和动态图表教程汇总
交互式图表和动态图表教程汇总

本专题整合了交互式图表和动态图表的相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.01.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
CSS3 教程
CSS3 教程

共18课时 | 4.5万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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