《C++实战项目-高并发内存池》7.大块内存的申请与释放

💡Yupureki:个人主页

✨个人专栏:《C++》 《算法》《Linux系统编程》《高并发内存池》


🌸Yupureki🌸的简介:


目录

[1. 大块内存的申请](#1. 大块内存的申请)

[2. 大块内存的释放](#2. 大块内存的释放)


完整项目链接https://github.com/Yupureki-code/ConcurrentMemoryPool

1. 大块内存的申请

ThreadCache和CentralCache内部能够存储的最大内存块为256kb。如果进程申请了超过256kb的内存块时,应该越级直接向PageCache申请

这个判断应该在最开始的向ThreadCache申请资源环节进行

cpp 复制代码
static void* ConcurrentAlloc(size_t size)
{
	if (size > MAX_BYTES)//大于256kb
	{
		size_t alignSize = SizeClass::RoundUp(size);
		size_t kpage = alignSize >> PAGE_SHIFT;

		PageCache::GetInstance()->_pageMtx.lock();
		Span* span = PageCache::GetInstance()->NewSpan(kpage);//直接向PageCache申请
		span->_objSize = size;
		PageCache::GetInstance()->_pageMtx.unlock();

		void* ptr = (void*)(span->_pageId << PAGE_SHIFT);
		return ptr;
	}
	else
	{
		// 通过TLS 每个线程无锁的获取自己的专属的ThreadCache对象
		if (pTLSThreadCache == nullptr)
		{
			static ObjectPool<ThreadCache> tcPool;
			//pTLSThreadCache = new ThreadCache;
			pTLSThreadCache = tcPool.New();
		}

		//cout << std::this_thread::get_id() << ":" << pTLSThreadCache << endl;

		return pTLSThreadCache->Allocate(size);
	}	
}

同时PageCache内部能够存储的最大页数为128,因此有两种情况

  1. 内存块大于256kb 小于 128 * 8 kb,则可以在PageCache内部拿
  2. 如果大于128 * 8 kb,那么说明PageCache也做不到,应该越级向系统申请

因此当PageCache收到这个请求后,也应该做特殊对待

cpp 复制代码
// 获取一个K页的span
Span* PageCache::NewSpan(size_t k)
{
	assert(k > 0);

	// 大于128 page的直接向堆申请
	if (k > NPAGES-1)
	{
		void* ptr = SystemAlloc(k);
		//Span* span = new Span;
		Span* span = _spanPool.New();

		span->_pageId = (PAGE_ID)ptr >> PAGE_SHIFT;
		span->_n = k;

		_idSpanMap[span->_pageId] = span;
		return span;
	}
    .......
}

2. 大块内存的释放

当进程释放这个大块内存后,原来是向PageCache申请的,现在也应该直接向PageCache释放

cpp 复制代码
static void ConcurrentFree(void* ptr)
{
	Span* span = PageCache::GetInstance()->MapObjectToSpan(ptr);
	size_t size = span->_objSize;

	if (size > MAX_BYTES)//释放大块内存
	{
		PageCache::GetInstance()->_pageMtx.lock();
		PageCache::GetInstance()->ReleaseSpanToPageCache(span);
		PageCache::GetInstance()->_pageMtx.unlock();
	}
	else
	{
		assert(pTLSThreadCache);
		pTLSThreadCache->Deallocate(ptr, size);
	}
}

同时PageCache最大只能存储128页的内存块,因此如果大于128 * 8 kb的大小,应该直接向系统释放

cpp 复制代码
void PageCache::ReleaseSpanToPageCache(Span* span)
{
	// 大于128 page的直接还给堆
	if (span->_n > NPAGES-1)
	{
		void* ptr = (void*)(span->_pageId << PAGE_SHIFT);
		SystemFree(ptr);
		//delete span;
		_spanPool.Delete(span);

		return;
	}
    ......
}
相关推荐
Vinton_Liu4 分钟前
NAT 类型详解:四种 NAT 的数据流与原理解析
运维·服务器
想唱rap6 分钟前
五种IO模型和非阻塞IO
linux·运维·服务器·网络·数据库·tcp/ip
周末也要写八哥10 分钟前
在C++中使用预定义宏
开发语言·c++·算法
Data_Journal19 分钟前
使用Python lxml轻松进行网络爬取
开发语言·php
xcLeigh33 分钟前
IoTDB JDBC 完整使用教程:连接、查询、批处理与字符集配置
开发语言·数据库·qt·iotdb·查询·批处理·连接
学会870上岸华师36 分钟前
C 语言程序设计——第一章课后编程题
c语言·开发语言·学习·算法
小小编程路37 分钟前
新手快速学 Python 极简速成指南
开发语言·c++·python
小马过河R44 分钟前
RAG检索优化策略:系统性四层框架解析
人工智能·python·算法·ai·llm·rag·问答
方安乐1 小时前
交换机的自学机制
运维·服务器·网络
AI技术控1 小时前
论文解读:AE-TCN-SA——基于自编码器、TCN 与自注意力机制的锂电池内短路诊断方法
人工智能·python·深度学习·算法·机器学习·自然语言处理