高并发的内存池

优势

  1. 内存碎片问题。(尽量缓解)
  2. 性能问题。(提高申请内存的效率)
  3. 多核多线程环境下,锁竞争问题

结构

在这里插入图片描述

threadCache

在这里插入图片描述
第一层的结构是一个

class ThreadCache
{
public:
	void* Allocate(size_t size);
	void Deallocate(void* ptr,size_t size);
	void* FetchFromCentralCache(size_t index, size_t size);

	void ListTooLong(FreeList* list, size_t size);

private:
	FreeList _freelist[NLISTS];
};
_declspec (thread)  static ThreadCache* tlslist = nullptr;

TLS技术:使每个线程都有一个tlslist
主要功能:

申请内存,有内存时直接从链表中拿取,没有的话从CentralCache中申请
释放内存:当在一个链表的后面挂的内存达到一定数量的时候,就归还给CentralCache

CentralCache

在这里插入图片描述
第二层的结构是:

class CentralCache
{
public:
	static CentralCache* GetInstance()
	{
		return &_inst;
	}

	//从Page Cache获取一个Span
	Span* GetOneSpan(SpanList& spanlist, size_t byte_sizes);
	
	//从中心缓存获取一定数量的对象给Thread Cache
	size_t FetchRangeObj(void*&start, void*&end, size_t n, size_t byte_sizes);

	//将一定数量的对象释放到span跨度
	void ReleaseListToSpan(void* start, size_t size);
private:
	SpanList _spanlist[NLISTS];
private:
	CentralCache()
	{}
	CentralCache(CentralCache&) = delete;
	static CentralCache _inst;
};

实现功能:

获取内存:当threadCache向CentralCache申请内存时,如果对应的链表上有内存的话,根据计算取出给threadCache,倘若没有,找PageCache申请
合并内存:当对应的span全部被释放的时候,就会进行合并算法,合成一块大内存,还给PageCache

PageCache

在这里插入图片描述
第三层的结构是:

class PageCache
{
public:
	static PageCache* GetInstance()
	{
		return &_inst;
	}
	Span* NewSpan(size_t size);

	Span* MapObjectToSpan(void* obj);

	void ReleaseSpanToPageCache(Span* span);

private:
	SpanList _spanlist[NPAGES];
	std::map<PageID, Span*> _idspanmap;
private:
	PageCache(){}
	PageCache(const PageCache&) = delete;
	static PageCache _inst;

};

功能:

申请内存:直接在共享区划分一部分内存,申请128页的,挂在对应的链表上
合并内存:合并对应的span,小的往大合并

性能

  • 加锁区间小,只有在对应的CentralCache的对应大小的链表上申请内存时,才需要加锁
  • 减少了内存碎片,控制在12%左右
  • 一次申请批量,提高了效率
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值