首页 > 系统相关 >定长内存池的实现

定长内存池的实现

时间:2024-06-01 14:31:43浏览次数:24  
标签:obj 实现 void 链表 内存 freeList size

定长内存池的实现


定长内存池只支持固定大小内存块的申请和释放

如何实现定长

  1. 使用非类型模板参数,使得内存池中申请的内存块大小都是N
template<size_t N>
class ObjectPool {};
  1. 定长内存池也叫做对象池,在创建对象池时,可以根据传入的对象类型的大小来实现“定长”,因此我们可以通过使用模板参数来实现“定长”
template<typename T>
class ObjectPool {};

如何直接向堆申请空间?

在Windows下,可以调用VirtualAlloc函数;在Linux下,可以调用brk或mmap函数。

void* SystemAlloc(size_t kpage) {
#ifdef _WIN32
	void* ptr = VituralAlloc(NULL, kpage << 13, MEM_COMMIT | MEM_RESERVE, PAGE_READWRITE);
#else
	// linux brk mmap
#endif
	if (ptr == nullptr) {
		throw std::bad_alloc();
	}
	return ptr;
}

#if 0
LPVOID VirtualAlloc(
  LPVOID lpAddress,     // 指定分配内存的起始地址,传入 NULL 由系统决定地址
  SIZE_T dwSize,       // 要分配的内存大小(字节)
  DWORD  flAllocationType, // 内存分配类型
  DWORD  flProtect      // 内存保护属性
);
#endif

定长内存池中应该包含哪些成员变量?

_memory:指向大块内存的指针。
_remainBytes:大块内存切分过程中剩余字节数。
_freeList:还回来过程中链接的自由链表的头指针。

内存池如何管理释放的对象?

对于还回来的定长内存块,我们可以用自由链表将其链接起来,但我们并不需要为其专门定义链式结构,我们可以让内存块的前4个字节(32位平台)或8个字节(64位平台)作为指针,存储后面内存块的起始地址
因此在向自由链表插入被释放的内存块时,先让该内存块的前4个字节或8个字节存储自由链表中第一个内存块的地址,然后再让_freeList指向该内存块即可,也就是一个简单的链表头插操作
Alt

如何在32/64位平台下访问到内存块的前4/8个字节?

  1. 首先,将原始指针ptr转换为指向指针的指针即(void**)ptr(32位下是4个字节、64位下是8个字节),接下来,通过解引用(void**)ptr 读取内存块的前4/8个字节。
void*& NextObj(void* ptr) {
	return *(void**)ptr;
}
  1. 使用联合体
union Header {
    void* asPointer;
    uint64_t as64Bit;
    uint32_t as32Bit;
};

void processMemoryBlock(void* ptr) {
    Header header;
    header.asPointer = ptr;
    
    // 根据平台自动选择访问大小
    #ifdef _WIN64 // 或其他宏定义来检测64位平台
        uint64_t first8Bytes = header.as64Bit;
    #else
        uint32_t first4Bytes = header.as32Bit;
    #endif
}

释放对象

我们使用的是placement new,那么需要我们显示调用对象的析构函数清理对象。

void Delete(T* obj) {
	obj->~T();
	//将释放的对象头插到自由链表
	NextObj(obj) = _freeList;
	_freeList = obj;
}

内存池如何为我们申请对象?

  • 当我们申请对象时,内存池应该优先把还回来的内存块对象再次重复利用,因此如果自由链表当中有内存块的话,就直接从自由链表头删一个内存块进行返回即可。
  • 如果自由链表当中没有内存块,那么就在大块内存中切出定长的内存块进行返回,当内存块切出后及时更新_memory指针的指向,以及_remainBytes的值即可。
T* New() {
	T* obj = nullptr;
	if (_freeList != nullptr) {
		obj = (T*)_freeList;
		_freeList = NextObj(_freeList);
	} else {
		// 限制最小
		size_t objSize = sizeof(T) < sizeof(void*) ? void(*) : sizeof(T);
		// 开辟空间
		if (_remainBytes < objSize) {
			_remainBytes = 128 * 1024;
			_memory = (char*)SystemAlloc(_remainBytes >> 13);
			if (_memory == nullptr) {
				throw std::bad_alloc();
			}
		}
		obj = (T*)_memory;
		_memory += objSize;
		_remainBytes -= objSize;
	}
	// placement new
	new (obj) T;
	return obj;
}

完整代码

//定长内存池
template<class T>
class ObjectPool
{
public:
	//申请对象
	T* New()
	{
		T* obj = nullptr;

		//优先把还回来的内存块对象,再次重复利用
		if (_freeList != nullptr)
		{
			//从自由链表头删一个对象
			obj = (T*)_freeList;
			_freeList = NextObj(_freeList);
		}
		else
		{
			//保证对象能够存储得下地址
			size_t objSize = sizeof(T) < sizeof(void*) ? sizeof(void*) : sizeof(T);
			//剩余内存不够一个对象大小时,则重新开大块空间
			if (_remainBytes < objSize)
			{
				_remainBytes = 128 * 1024;
				//_memory = (char*)malloc(_remainBytes);
				_memory = (char*)SystemAlloc(_remainBytes >> 13);
				if (_memory == nullptr)
				{
					throw std::bad_alloc();
				}
			}
			//从大块内存中切出objSize字节的内存
			obj = (T*)_memory;
			_memory += objSize;
			_remainBytes -= objSize;
		}
		//定位new,显示调用T的构造函数初始化
		new(obj)T;

		return obj;
	}
	//释放对象
	void Delete(T* obj)
	{
		//显示调用T的析构函数清理对象
		obj->~T();

		//将释放的对象头插到自由链表
		NextObj(obj) = _freeList;
		_freeList = obj;
	}
private:
	char* _memory = nullptr;     //指向大块内存的指针
	size_t _remainBytes = 0;     //大块内存在切分过程中剩余字节数

	void* _freeList = nullptr;   //还回来过程中链接的自由链表的头指针
};

性能测试

#include "../src/ObjectPool.cpp"
#include "cstddef"
#include <vector>
#include <iostream>

using namespace std;

struct TreeNode
{
	int _val;
	TreeNode* _left;
	TreeNode* _right;
	TreeNode()
		:_val(0)
		, _left(nullptr)
		, _right(nullptr)
	{}
};

void TestObjectPool()
{
	// 申请释放的轮次
	const size_t Rounds = 3;
	// 每轮申请释放多少次
	const size_t N = 1000000;
	std::vector<TreeNode*> v1;
	v1.reserve(N);

	//malloc和free
	size_t begin1 = clock();
	for (size_t j = 0; j < Rounds; ++j)
	{
		for (int i = 0; i < N; ++i)
		{
			v1.push_back(new TreeNode);
		}
		for (int i = 0; i < N; ++i)
		{
			delete v1[i];
		}
		v1.clear();
	}
	size_t end1 = clock();

	//定长内存池
	ObjectPool<TreeNode> TNPool;
	std::vector<TreeNode*> v2;
	v2.reserve(N);
	size_t begin2 = clock();
	for (size_t j = 0; j < Rounds; ++j)
	{
		for (int i = 0; i < N; ++i)
		{
			v2.push_back(TNPool.New());
		}
		for (int i = 0; i < N; ++i)
		{
			TNPool.Delete(v2[i]);
		}
		v2.clear();
	}
	size_t end2 = clock();

	cout << "new cost time:" << end1 - begin1 << endl;
	cout << "object pool cost time:" << end2 - begin2 << endl;
}

int main() {
    TestObjectPool();
    return 0;
}

经过测试性能大概提升3倍!

标签:obj,实现,void,链表,内存,freeList,size
From: https://blog.csdn.net/weixin_51332735/article/details/139241304

相关文章

  • Axios请求失败重试实现
    代码实现//request.tsimportaxios,{AxiosRequestConfig}from"axios";//自定义ReuqestConfiginterfaceRetryConfgextendsAxiosRequestConfig{retry:number,retryDelay:number,retryCount?:number;}//创建请求实例constrequest=axios.......
  • Windows Server 2008实现磁盘管理
    一、认识磁盘管理和各种磁盘磁盘类型        基本磁盘:时间久、应用广泛的一种磁盘类型,兼容性好,兼容微软所有的操作系统,磁盘分区,包括主分区/扩展分区/逻辑分区        动态磁盘:Windows2000/2003/XP支持,比基本磁盘具有更好的扩展性和可靠性磁盘管理工具(1)......
  • 通过Restful接口实现对数据库进行基本的读写操作
    一、创建springboot项目这里使用的springboot项目与我们上次使用的一样,所以创建方法跟下文一致即可。SpringBoot+MySQL的简单运用(HelloWorldAPI)-CSDN博客二、配置数据库连接2.1创建数据库首先我们要创建一个数据库,应用于本次项目。打开MySQL,输入密码登录。(MySQL安......
  • 使用Spring Boot自定义注解 + AOP实现基于IP的接口限流和黑白名单
    ......
  • HTML期末学生大作业-基于班级校园我的校园网页设计与实现html+css+javascript
    ......
  • 单链表实现通讯录
    之前我们完成了基于顺序表(动态)实现通讯录,现在我们链表学完了,可以尝试着使用链表来实现我们的通讯录。首先我们要明白我们写的通讯录是由一个个节点组成的,每个节点里存储的就是我们的联系人信息。也就是说我们需要先写一个单链表,完成单链表的插入,删除等功能。然后在单链表......
  • 基于单片机的汽车防盗报警系统设计与实现
    摘要:为了有效保护车辆,防止车辆被盗,汽车防盗报警系统的设计成为研究的热点问题。基于STC89C52单片机设计了一套汽车防盗报警系统,该系统由硬件和软件两部分组成,通过高效集成电路形成完整的控制系统,电路结构简单、易于检测。介绍了各部件的结构及工作原理,利用实车试验对......
  • 构建无与伦比的深度学习环境:在CentOS上实现GPU资源管理容器的终极指南
    在CentOS上构建无与伦比的深度学习环境并实现GPU资源管理容器的过程,可以概括为以下关键步骤:一、安装NVIDIA驱动首先,为了使用GPU进行深度学习训练,你需要安装NVIDIA驱动。这通常包括以下步骤:导入ELRepo仓库的GPG密钥:bash复制代码rpm--importhttps://www.elrepo.org......
  • ESP32-C3模组上实现蓝牙BLE配网功能(1)
    本文内容参考:《ESP32-C3物联网工程开发实战》乐鑫科技蓝牙的名字由来是怎样的?为什么不叫它“白牙”?特此致谢!一、蓝牙知识基础1.什么是蓝牙?(1)简介蓝牙技术是一种无线数据和语音通信开放的全球规范,它是基于低成本的近距离无线连接,为固定和移动设备建立通信环境的一种特......
  • 用 pytorch 从零开始实现单隐层 MLP
    我的代码如下:importtorchfromtorchvisionimporttransformsfromtorch.utilsimportdata导入torchvision#==============load数据集defget_dataloader_workers():返回4defload_data_fashion_mnist(batch_size,resize=None):trans=[transforms.ToT......