0

0

如何在c++中实现一个高性能的对象池 (Object Pool)? (内存复用)

冰火之心

冰火之心

发布时间:2026-01-13 07:51:13

|

745人浏览过

|

来源于php中文网

原创

std::vector + std::optional 不适合高频对象池,因其每次 emplace/ reset 都触发构造/析构,违背绕过生命周期开销的核心目标;高性能池须用 std::aligned_storage_t(或 C++23 的 std::byte[])手动管理裸内存,结合 placement-new 与显式析构,并采用 thread_local 子池+无锁回收机制。

如何在c++中实现一个高性能的对象池 (object pool)? (内存复用)

为什么 std::vector + std::optional 不适合高频对象池?

直接用 std::vector<:optional>> 管理闲置对象,看似简单,但每次 std::optional::emplace() 都会触发构造,reset() 触发析构 —— 这违背了对象池“绕过构造/析构开销”的核心目标。真正高性能的池必须分离内存分配与对象生命周期管理。

std::aligned_storage_t 手动管理原始内存

关键不是“存对象”,而是“存一块对齐、足量、可复用的裸内存”。对象只在 acquire() 时用 placement-new 构造,在 release() 时显式调用析构函数,不释放内存。

  • std::aligned_storage_t 提供类型无关的对齐内存块
  • std::vector 存储这些存储单元(避免频繁系统调用)
  • std::stack 维护空闲索引,O(1) 分配/回收
  • 禁止拷贝,只支持移动;所有成员变量需 mutable 或用指针缓存状态(如是否已构造)
template
class ObjectPool {
    std::vector> storage_;
    std::stack free_list_;
public:
    explicit ObjectPool(size_t initial_size = 1024) : storage_(initial_size) {
        for (size_t i = 0; i < initial_size; ++i) free_list_.push(i);
    }
T* acquire() {
    if (free_list_.empty()) {
        storage_.emplace_back();
        free_list_.push(storage_.size() - 1);
    }
    size_t idx = free_list_.top();
    free_list_.pop();
    return new (&storage_[idx]) T(); // placement-new
}

void release(T* obj) {
    obj->~T(); // 显式析构
    free_list_.push(static_cast(obj - reinterpret_cast(storage_.data())));
}

};

如何避免虚析构和 RTTI 带来的开销?

如果池中对象有虚函数,且你通过基类指针 release(),编译器可能插入 RTTI 查找实际类型 —— 这破坏确定性延迟。解决方案只有一个:池必须与具体类型强绑定。

Kubit.ai
Kubit.ai

一个AI驱动的产品分析平台,为产品和数据团队构建

下载

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

  • 不要写 ObjectPool 并往里塞 Derived 实例
  • 每个具体类型(如 ConnectionPacket)单独实例化池
  • 若需多态行为,用组合代替继承:池内对象持有 std::function 或函数指针,而非继承体系
  • 确保 T 满足 std::is_trivially_destructible_v 时跳过 obj->~T() 可进一步减小分支开销

线程安全与内存顺序怎么加才不拖慢性能?

全局锁会让高并发场景下所有线程排队,吞吐骤降。更优解是每个线程独占一个子池(thread_local),再配合中央池做跨线程回收。

  • 主线程或专用回收线程定期合并各 thread_local 池的空闲块到共享池
  • 避免在 acquire() 路径上用 std::atomic 修改计数器;改用无锁(如 boost::lockfree::stack)或分段锁
  • 注意 std::aligned_storage_t 在 C++23 中已被弃用,应迁移到 std::byte[] + std::assume_aligned,但目前主流编译器仍需兼容前者

最易被忽略的一点:对象池的“高性能”永远依赖使用方严格遵守 acquire/release 成对原则。漏掉一次 release,就等于内存泄漏;多调一次,就是未定义行为 —— 这种错误不会报错,只会随机崩溃或静默数据污染。

相关专题

更多
java多态详细介绍
java多态详细介绍

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

15

2025.11.27

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

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

389

2023.07.18

堆和栈区别
堆和栈区别

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

571

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

480

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

480

2023.08.10

function是什么
function是什么

function是函数的意思,是一段具有特定功能的可重复使用的代码块,是程序的基本组成单元之一,可以接受输入参数,执行特定的操作,并返回结果。本专题为大家提供function是什么的相关的文章、下载、课程内容,供大家免费下载体验。

476

2023.08.04

js函数function用法
js函数function用法

js函数function用法有:1、声明函数;2、调用函数;3、函数参数;4、函数返回值;5、匿名函数;6、函数作为参数;7、函数作用域;8、递归函数。本专题提供js函数function用法的相关文章内容,大家可以免费阅读。

163

2023.10.07

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

4

2026.01.15

公务员递补名单公布时间 公务员递补要求
公务员递补名单公布时间 公务员递补要求

公务员递补名单公布时间不固定,通常在面试前,由招录单位(如国家知识产权局、海关等)发布,依据是原入围考生放弃资格,会按笔试成绩从高到低递补,递补考生需按公告要求限时确认并提交材料,及时参加面试/体检等后续环节。要求核心是按招录单位公告及时响应、提交材料(确认书、资格复审材料)并准时参加面试。

23

2026.01.15

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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