抽象数据结构
如图所示,pool模块核心数据结构之间的关系。
一个caching_pool 可以创建和管理多个 pool。
一个pool可以申请创建多个block内存空间。
关于内存申请
于是就很清楚pool相关接口的调用流程了
关于释放
需要注意的是:
1) 使用 pj_pool_alloc 申请的内存是没有对应的函数释放的, 只有使用pj_pool_release函数释放pool时才能被全部释放。
2)如果有一些逻辑需要频繁使用pj_pool_alloc申请内存,但是很快这些内存就不用了,也无法调用pj_pool_release释放pool。 这时候应该对这些频繁申请内存的逻辑创建一个新的pool。避免内存泄漏。pool 管理不善也会出现内存泄漏。
3)有时候使用 pj_pool_reset 比使用 pj_pool_release更好。
pj_pool_reset 不会再系统层面释放内存,而是将其 所属的 block指针还原。
再次使用 pj_pool_alloc时内存可以被重新分配。原来的pool指针还是可用的。
pj_pool_release 之后, pool整个被释放,pool指针将不可用。
pool_alt
pool_alt.h
pool_dbg.c
添加了更多调试打印,需要 在#include <pj/pool.h> 前面 配置 #define PJ_HAS_POOL_ALT_API 才能启用。方便调试。
pj_pool_create_on_buf
pool_buf.h
pool_buf.c
PJ_DECL(pj_pool_t*) pj_pool_create_on_buf(const char *name,
void *buf,
pj_size_t size);
我们之前pjlib中其他模块往往需要依赖pj_pool_t. 正常pool是从堆中申请内存,申请释放开销是比较大。
如果我们只是在一个函数中用到某个需要pool 的模块,函数退出就释放。那么可以使用该函数临时使用pj_pool_t. 不需要初始化caching_pool和释放。
#include <pjlib.h>
static void test()
{
char buffer[500];
pj_pool_t *pool;
void *p;
pool = pj_pool_create_on_buf("thepool", buffer, sizeof(buffer));
// Use the pool as usual
p = pj_pool_alloc(pool, ...);
...
// No need to release the pool
}
int main()
{
pj_init();
test();
return 0;
}