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 查找实际类型 —— 这破坏确定性延迟。解决方案只有一个:池必须与具体类型强绑定。

SEO GPT
SEO GPT

免费的白帽SEO,PPC和网站经销商平台

下载

立即学习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,就等于内存泄漏;多调一次,就是未定义行为 —— 这种错误不会报错,只会随机崩溃或静默数据污染。

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

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

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

15

2025.11.27

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

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

386

2023.07.18

堆和栈区别
堆和栈区别

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

569

2023.08.10

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

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

480

2023.08.10

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

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

480

2023.08.10

function是什么
function是什么

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

474

2023.08.04

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

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

163

2023.10.07

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

9

2026.01.12

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

101

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外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号