对象池的核心目标是复用资源,减少频繁创建和销毁的开销。1. 通过定制删除器,shared_ptr 在引用计数归零时不释放内存而是将对象放回池中;2. 对象池结构包含存储容器、删除器、获取和释放方法;3. 需注意避免裸指针误删、线程安全、池大小限制及构造参数支持等细节;4. 实现方式兼顾安全与性能,适合高频场景但需处理状态重置和并发问题。

对象池的核心目标是复用资源,减少频繁创建和销毁的开销。用 C++ 的 shared_ptr 实现对象池时,关键在于定制删除器:当引用计数归零时,不是直接释放内存,而是把对象“放回池中”。

为什么需要定制删除器?
默认情况下,shared_ptr 在最后一个引用失效时会调用 delete。但对象池希望这个对象不被真正删除,而是回收到池里等待下次使用。

这时候就需要一个自定义的删除器函数(或仿函数),告诉 shared_ptr:
“别删了它,把它还给我。”
这样就能控制对象生命周期,实现复用。
如何设计对象池结构?
一个简单的对象池可以包含以下几个部分:
- 存储可用对象的容器(如
std::stack) - 定制删除器函数
- 获取和释放对象的方法
举个例子:
templateclass ObjectPool { public: using Deleter = std::function ; // 池内回收对象的删除器 static void returnToPool(T* ptr) { instance()._pool.push(ptr); } // 获取一个对象,如果池里有就复用,没有就新建 shared_ptr get() { if (_pool.empty()) { return shared_ptr (new T, returnToPool); } else { T* obj = _pool.top(); _pool.pop(); return shared_ptr (obj, returnToPool); } } private: std::stack _pool; static ObjectPool& instance() { static ObjectPool pool; return pool; } };
这种方式让每次获取的对象都带上了“回收逻辑”,只要没人用了,就会自动回到池里。
使用时需要注意哪些细节?
-
避免裸指针误删
- 不要对外暴露原始指针,只返回
shared_ptr,否则用户可能手动 delete 导致崩溃。
- 不要对外暴露原始指针,只返回
-
线程安全问题
- 如果多个线程同时访问对象池,
stack和shared_ptr的操作都需要加锁。 - 可以在
get()和删除器中加互斥锁保护_pool的访问。
- 如果多个线程同时访问对象池,
-
池的大小限制
- 可以设置最大容量,超过上限再释放内存,而不是无限制增长。
- 也可以定期清理空闲对象,节省资源。
-
构造参数支持
- 上面的例子只能默认构造对象,若想传递参数,可以用工厂方法封装
new T(...)。
- 上面的例子只能默认构造对象,若想传递参数,可以用工厂方法封装
这样做有什么好处和限制?
优点:
- 管理方便:靠智能指针自动触发回收逻辑,不用手动跟踪谁在用哪个对象。
- 性能提升:减少动态分配次数,适合高频创建销毁的场景,比如网络连接、数据库连接等。
缺点:
- 实现复杂一些,特别是要考虑线程安全。
- 对象状态不清空,复用前需要重置,否则可能出现残留数据影响新用途。
基本上就这些。用 shared_ptr 加定制删除器来实现对象池,是一种兼顾安全和性能的做法,虽然细节上要注意点,但一旦搭好,后续使用很省心。










