C++实现简单的固定大小内存分配器

2023-11-18

背景介绍

最近在了解性能优化相关的知识,了解到内存分配是一个可优化性能的方向,在相关书籍中看到一个固定大小的内存分配器实现,觉得挺有意思的,这里跟大家分享, 据说比通用的内存分配malloc()性能有不少提升。

源码实现

这里的代码是根据《c性能优化指南》中第13章编写的,详细的代码解析在书中有,这里就不赘述了,大家感兴趣可以瞅一瞅原文内容。

#include <iostream>
#include <algorithm>

/*memory allocator*/
struct fixed_arena_controller
{
	template<int N>
	fixed_arena_controller(char(&a)[N]);

	fixed_arena_controller(fixed_arena_controller&) = delete;
	~fixed_arena_controller() = default;
	void operator=(fixed_arena_controller&) = delete;

	void* allocate(size_t);
	size_t block_size() const;
	size_t capacity() const;
	bool empty() const;
	void clear();

private:
	void* arena_;
	size_t arena_size_;
	size_t block_size_;
};

template<int N>
inline fixed_arena_controller::fixed_arena_controller(char(&a)[N]) :
	arena_(a), arena_size_(N), block_size_(0) {}

inline void* fixed_arena_controller::allocate(size_t size)
{
	if (!empty()) {
		return nullptr;
	}

	block_size_ = std::max(size, sizeof(void*));
	size_t count = capacity();

	if (count == 0) {
		return nullptr;
	}

	char* p;
	for (p = (char*)arena_; count > 1; --count, p += size) {
		*reinterpret_cast<char**>(p) = p + size;
	}
	*reinterpret_cast<char**>(p) = nullptr;
	return arena_;
}

inline size_t fixed_arena_controller::block_size() const
{
	return block_size_;
}

inline size_t fixed_arena_controller::capacity() const
{
	return block_size_ ? (arena_size_ / block_size_) : 0;
}

inline void fixed_arena_controller::clear()
{
	block_size_ = 0;
}

inline bool fixed_arena_controller::empty() const
{
	return block_size_ == 0;
}


/*memory manager*/
template<class Arena>
struct fixed_block_memory_manager
{
	template<int N>
	fixed_block_memory_manager(char(&a)[N]);

	fixed_block_memory_manager(fixed_block_memory_manager&) = delete;
	~fixed_block_memory_manager() = default;
	void operator=(fixed_block_memory_manager&) = delete;

	void* allocate(size_t);
	size_t block_size() const;
	size_t capacity() const;
	void clear();
	void deallocate(void*);
	bool empty() const;

private:
	struct free_block
	{
		free_block* next;
	};
	free_block* free_ptr_;
	size_t		block_size_;
	Arena		arena_;
};

template<class Arena>
	template<int N>
		inline fixed_block_memory_manager<Arena>
			::fixed_block_memory_manager(char(&a)[N]) :
			arena_(a), free_ptr_(nullptr), block_size_(0)
		{}

template<class Arena>
	inline void* fixed_block_memory_manager<Arena>
		::allocate(size_t size)
{
	if (empty()) {
		free_ptr_ = reinterpret_cast<free_block*>
			(arena_.allocate(size));
		block_size_ = size;
		if (empty()) {
			throw std::bad_alloc();
		}
	}

	if (size != block_size_) {
		throw std::bad_alloc();
	}

	auto p = free_ptr_;
	free_ptr_ = free_ptr_->next;
	return p;
}

template<class Arena>
	inline void fixed_block_memory_manager<Arena>
		::deallocate(void* p)
{
	if (p == nullptr) {
		return;
	}

	auto fp = reinterpret_cast<free_block*>(p);
	fp->next = free_ptr_;
	free_ptr_ = fp;
}

template<class Arena>
	inline size_t fixed_block_memory_manager<Arena>
		::capacity() const
	{
		return arena_.capacity();
	}

template<class Arena>
	inline void fixed_block_memory_manager<Arena>
		::clear()
	{
		free_ptr_ = nullptr;
		arena_.clear();
	}

template<class Arena>
	inline bool fixed_block_memory_manager<Arena>
		::empty() const
	{
		return arena_.empty();
	}

/*operator new() for class*/
class MemMgrTester
{
private:
	int contents_;

public:
	static fixed_block_memory_manager<fixed_arena_controller> mgr_;

	MemMgrTester(int c) : contents_(c) {}

	static void* operator new(size_t s)
	{
		std::cout << "new..." << std::endl;
		return mgr_.allocate(s);
	}

	static void operator delete(void* p)
	{
		std::cout << "delete..." << std::endl;
		mgr_.deallocate(p);
	}
};

char arena[4004];
fixed_block_memory_manager<fixed_arena_controller> MemMgrTester::mgr_(arena);

int main()
{
	MemMgrTester* pObj = new MemMgrTester(100);
	delete pObj;
		 
	system("pause");
	return 0;
}

参考书籍

《C性能优化指南》

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++实现简单的固定大小内存分配器 的相关文章

随机推荐

  • H5打开新窗口与页面跳转的几种方法

    1 在本窗体打开一个新的页面 window location href URL 2 在一个新的窗口打开一个新的页面 window open URL 3 本窗口的页面被替换为一个新的页面URL 替换后不可以回退到上个页面 location r
  • 理光打印机出现sc542-01怎样解决?

    问题图 步骤一图文 补充 理光维修模式指令 107 03以下机型 143 通用 806182 03以上机型 注方法 1 清除键 黄色 点一次 107 停止键 红色键 按住3秒以上 2 依次按重置 RESET 806182 按住C不动3秒 就
  • CentOS7.5最小化安装之后的配置

    我是最小化安装的 安装了之后很多基本使用配置没有 接下来要做一些配置 如网络之类的 使系统可用 1 使命令分页显示 1页显示不不下 又不能上翻页 xxx more 2 查看系统安装了哪些软件包 rpm qa 3 配置网络 最小化安装之后 i
  • 听我一句劝,别去外包,干了3个月,技术退步明显...

    先说一下自己的情况 大专生 18年通过校招进入湖南某软件公司 干了接近4年的功能测试 今年年初 感觉自己不能够在这样下去了 长时间呆在一个舒适的环境会让一个人堕落 而我已经在一个企业干了四年的功能测试 已经让我变得不思进取 谈了2年的女朋友
  • 一种关于单片机定时器中断和数码管冲突问题的解决方案

    问题发现 我们会发现 同时存在定时器中断和数码管操作时 有时会导致数码管显示异常 原因探究 在定时器中断函数中不要操作P2和P0 因为定时器 T 和主板 M 的时钟频率不一样 有可能导致M刚操作完P2 T又去操作P0 导致正确的P2和P0没
  • Windows从命令行创建文本文件的两种方式

    第一种方式 copy con 文件名 然后可以输入文件内容 完成后打Ctrl Z退出 输入的内容被保存为一个文件 第二种 如下 echo gt 文件名 创建一个空的文件 当然你可以创建任何后缀名的文本类文件 创建一个SQL文件也是没问题的
  • 教你更优雅的写法处理null检查

    回顾以前对null的处理方式 public class OptionalTest public static void main String args test01 小黑 test01 null public static void te
  • View.VISIBLE、INVISIBLE、GONE的区别

    View VISIBLE INVISIBLE GONE的区别 android中UI应用的开发中经常会使用view setVisibility 来设置控件的可见性 其中该函数有3个可选值 他们有着不同的含义 View VISIBLE gt 可
  • Qt事件和事件循环

    在处理QT循环事件的时候遇到了问题 查了半天资料都没弄明白问题出在哪 后来找大牛同事问了一下 同事就给我写了QCoreApplication processEvent 这个函数 好啦 终于搞定了 这里小记一下 以免以后遇到 于是乎这里认真仔
  • Your account has been blocked问题解决

    更新代码时报错 如下 branches to build service 1 get branch service to dir service only update code Your account has been blocked
  • 2020规划调整,以自研引擎和osg/osgearth源码分析为主

    随着公司转入webgl 我要求调整到osg osgearth工具 因为是c 方面的 特别是osg osgearth源码可以深入下 以前光把视频教程看过 说白了就是简单二次应用 并没有深入下去 所以 趁此机会多读读osg osgearth源码
  • 计算机系统(一)——冯诺依曼体系

    现代计算机的基本结构是由冯诺依曼提出的 冯诺依曼的体系结构体系如下图所示 计算机由运算器 控制器 存储器 也就是内存 输入设备 输出设备五大部件组成 其中运算器和控制器组成cpu 在冯诺依曼体系中 数据先从输入设备输入到内存中 运算时数据从
  • 解决使用SpringBoot上传文件大小受限问题

    今天在做项目学习的过程中 需要用到文件上传 启动项目正常 访问上传post请求突然报出了一个异常 特此记录一下 1 控制台异常 org springframework web multipart MaxUploadSizeExceededE
  • 一起赚美元①

    译者 一只椰子 知识星球链接 https t zsxq com iAiIyfm 本文的主人公叫迈克 他在采访里面提到 自动化工具这一法宝已经轻松帮他实现每个月 12 5万美元的利润 对他而言 年入100万美元 根本不是梦 1 主人公简介 我
  • 看完这篇,成为Grafana高手!

    背景 QQ直播前端团队接入腾讯云前端性能监控 RUM 后 对目前的监控能力以及上报数据进行了梳理 并着手进行了前端性能监控的专项建设 其中监控数据大盘建设是不可或缺的一环 可视化的监控大盘可以清晰明了的观察到各项目运行情况 宏观上能快速进行
  • 【ESP32S2】使用ESP32S2开发板制作U盘,支持无线访问

    文章目录 前言 一 开发环境 二 硬件介绍 三 下载esp idf 四 下载esp iot solution 五 配置idf 1 执行下载脚本 2 设置环境变量 3 声明外部组件 六 构建例程 七 烧写Flash 八 使用方法 九 感谢 前
  • 深度学习之Python,OpenCV中的卷积

    这篇博客将介绍图像内核和卷积 如果将图像视为一个大矩阵 那么图像内核只是一个位于图像顶部的微小矩阵 从左到右和从上到下滑动内核 计算输入图像和内核之间的元素乘法总和 称这个值为内核输出 内核输出存储在与输入图像相同 x y 坐标的输出图像中
  • 3D游戏第九次作业

    3D游戏第九次作业 1 血条 Health Bar 的预制设计 具体要求如下 分别使用 IMGUI 和 UGUI 实现 使用 UGUI 血条是游戏对象的一个子元素 任何时候需要面对主摄像机 分析两种实现的优缺点 给出预制的使用方法 IMGU
  • 2、为什么选择URP

    为什么选择URP 现在所在项目为SLG类项目 有同屏200 英雄的需求 Unity传统渲染管线的动态批处理有个限制是Skined Mesh Renderer不能进行动态批处理 英雄如果不采用Skined Mesh Renderer又不能满足
  • C++实现简单的固定大小内存分配器

    背景介绍 最近在了解性能优化相关的知识 了解到内存分配是一个可优化性能的方向 在相关书籍中看到一个固定大小的内存分配器实现 觉得挺有意思的 这里跟大家分享 据说比通用的内存分配malloc 性能有不少提升 源码实现 这里的代码是根据 c性能