我的编程空间,编程开发者的网络收藏夹
学习永远不晚

C++中怎么手动创建一个内存池

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

C++中怎么手动创建一个内存池

今天就跟大家聊聊有关C++中怎么手动创建一个内存池,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。

引言

使用new expression为类的多个实例分配动态内存时,cookie导致内存利用率可能不高,此时我们通过实现类的内存池来降低overhead。从不成熟到巧妙优化的内存池,得益于union的分时复用特性,内存利用率得到了提高。

原因

在实例化某个类的对象时(在heap而不是stack中),若不使用array new,则每次实例化时都要调用一次内存分配函数,类的每个实例在内存中都有上下两个cookie,从而降低了内存的利用率。然而,array new也有先天的缺陷,即只能调用默认无参构造函数,这对于很多没有提供无参构造函数的类来说是不合适的。

因此,我们可以对于一个没有实例化的类第一次实例化时,先分配一大块内存(内存池),这一大块内存记录在类中,只有上下两个cookie,能够容纳多个实例。后续实例化时,若内存池中还有剩余内存,则不必申请内存分配,只在内存池中分配。内存回收时,将实例所占用的内存回收到内存池中。若内存池中无内存,则再申请分配大块内存。

脱裤子放屁方案

我们以链表的形式组织内存池,内存池中每个一个链表是一个小桶,这个桶中装我们实例化的对象。

内存池链表的头结点记录在类中,即以class staic变量的形式存储。组织形式如下:

C++中怎么手动创建一个内存池

实现代码如下:

#include <iostream>using namespace std;class DemoClass{public:    DemoClass() = default;    DemoClass(int i):data(i){}    static void* operator new(size_t size);    static void operator delete(void *);    virtual ~DemoClass(){}private:    DemoClass *next;    int data;    static DemoClass *freeMemHeader;    static const size_t POOL_SIZE;};DemoClass * DemoClass::freeMemHeader = nullptr;const size_t DemoClass::POOL_SIZE = 24;//设定内存池能容纳24个DemoClass对象void* DemoClass::operator new(size_t size){    DemoClass* p;    if(!freeMemHeader){//freeMemHeader为空,内存池中无空间,分配内存        size_t pool_mem_bytes = size * POOL_SIZE;//内存池的字节大小 = 每个实例的大小(字节数)* 内存池中能容纳的最大实例数        freeMemHeader = reinterpret_cast<DemoClass*>(new char[pool_mem_bytes]);//new char[]分配pool_mem_bytes个字节,因为每个char占用1个字节        cout << "Info:向操作系统申请了" << pool_mem_bytes << "字节的内存。" << endl;        for(int i = 0;i < POOL_SIZE - 1; ++i){//将内存池中POOL_SIZE个小块内存,串起来。            freeMemHeader[i].next = &freeMemHeader[i + 1];        }        freeMemHeader[POOL_SIZE - 1].next = nullptr;    }    p = freeMemHeader;//取内存池(链表)的头部,分配给要实例化的对象    cout << "Info:从内存池中取了" << size << "字节的内存。" << endl;    freeMemHeader = freeMemHeader -> next;//从内存池中删去取出的那一小块地址,即更新内存池    p -> next = nullptr;    return p;}void DemoClass::operator delete(void* p){    DemoClass* tmp = (DemoClass*) p;    tmp -> next = freeMemHeader;    freeMemHeader = tmp;}

测试代码如下:

int main(int argc, char* argv[]){    cout << "sizeof(DemoClass):" << sizeof(DemoClass) << endl;    size_t N = 32;    DemoClass* demos[N];    for(int i = 0; i < N; ++i){        demos[i] = new DemoClass(i);        cout << "address of the ith demo:" << demos[i] << endl;        cout << endl;    }    return 0;}

其结果如下:

C++中怎么手动创建一个内存池

C++中怎么手动创建一个内存池

可以看到每个DemoClass的实例大小为24字节,内存池一次从操作系统中申请了576个字节的内存,这些内存可以容纳24个实例。上面显示出了每个实例的内存地址,内存池中相邻实例的内存首地址之差为24,即实例的大小,证明了一个内存池的实例之间确实没有cookie。

当内存池中内存用完后,又向操作系统申请了576个字节的内存。

由此,只有每个内存池两侧有cookie,而内存池中的实例不存在cookie,相比于每次调用new expression实例化对象都有cookie,内存池的组织形式确实在形式上提高了内存利用率。

那么,有什么问题么?

sizeof(DemoClass)等于24

  1. int data数据域占4个字节

  2. 两个构造函数一个析构函数各占4字节,共12字节

  3. 额外的指针DemoClass*,在64位机器上,占8个字节

这样一个DemoClass的大小确实是24字节。wait,what?

我们为了解决cookie带来的内存浪费,引入了指针next,但却又引入了8个字节的overhead,脱裤子放屁,多此一举?

这样看来确实没有达到要求,但至少为我们提供了一种思路,不是么?

分时复用改进方案

首先我们先回忆下c++ 中的Union:

在任意时刻,联合中只能有一个数据成员可以有值。当给联合中某个成员赋值之后,该联合中的其它成员就变成未定义状态了。

结合我们之前不成熟的内存池,我们发现,当内存池中的桶还没有被分配给实例时,只有next域有用,而当桶被分配给实例后,next域就没什么用了;当桶被回收时,数据域变无用而next指针又需要用到。这不正是union的特性么?

看一下代码实现:

#include <iostream>using namespace std;class DemoClass{public:    DemoClass() = default;    DemoClass(int i, double p){        data.num = i;        data.price = p;    }    static void* operator new(size_t size);    static void operator delete(void *);    virtual ~DemoClass(){}private:    struct DemoData{        int num;        double price;    };private:    static DemoClass *freeMemHeader;    static const size_t POOL_SIZE;    union {        DemoClass *next;        DemoData data;    };    };DemoClass * DemoClass::freeMemHeader = nullptr;const size_t DemoClass::POOL_SIZE = 24;//设定内存池能容纳24个DemoClass对象void* DemoClass::operator new(size_t size){    DemoClass* p;    if(!freeMemHeader){//freeMemHeader为空,内存池中无空间,分配内存        size_t pool_mem_bytes = size * POOL_SIZE;//内存池的字节大小 = 每个实例的大小(字节数)* 内存池中能容纳的最大实例数        freeMemHeader = reinterpret_cast<DemoClass*>(new char[pool_mem_bytes]);//new char[]分配pool_mem_bytes个字节,因为每个char占用1个字节        cout << "Info:向操作系统申请了" << pool_mem_bytes << "字节的内存。" << endl;        for(int i = 0;i < POOL_SIZE - 1; ++i){//将内存池中POOL_SIZE个小块内存,串起来。            freeMemHeader[i].next = &freeMemHeader[i + 1];        }        freeMemHeader[POOL_SIZE - 1].next = nullptr;    }    p = freeMemHeader;//取内存池(链表)的头部,分配给要实例化的对象    cout << "Info:从内存池中取了" << size << "字节的内存。" << endl;    freeMemHeader = freeMemHeader -> next;//从内存池中删去取出的那一小块地址,即更新内存池    p -> next = nullptr;    return p;}void DemoClass::operator delete(void* p){    DemoClass* tmp = (DemoClass*) p;    tmp -> next = freeMemHeader;    freeMemHeader = tmp;}

对比前一种实现代码,只是构造函数、数据域和指针域的组织形式发生了变化:

  • 由于数据域增加了price项,构造函数中也增加了对应的参数

  • 数据域被集成定义成一个类自定义struct类型

  • 数据域和指针域被组织为union

测试代码依旧:

int main(int argc, char* argv[]){    cout << "sizeof(DemoClass):" << sizeof(DemoClass) << endl;    size_t N = 32;    DemoClass* demos[N];    for(int i = 0; i < N; ++i){        demos[i] = new DemoClass(i, i * i);        cout << "address of the " << i << "th demo:" << demos[i] << endl;        cout << endl;    }    return 0;}

结果:

C++中怎么手动创建一个内存池

C++中怎么手动创建一个内存池

可以看到每个DemoClass的实例大小为24字节,一个内存池的实例之间没有cookie。

分析一下sizeof(DemoClass)等于24的缘由:

  • data数据域占12个字节(int 4字节、double 8字节)。

  • 两个构造函数一个析构函数各占4字节,共12字节。

  • 指针DemoClass,在64位机器上,占8个字节,但由于和数据域使用了union,data数据域12个字节中的前8个字节在适当的时机被看作DemoClass,而不占用额外空间,消除了overhead。

这样一个DemoClass的大小确实是24字节。利用union的分时复用特性,我们消除了初步方案中指针带来的脱裤子放屁效果。

看完上述内容,你们对C++中怎么手动创建一个内存池有进一步的了解吗?如果还想了解更多知识或者相关内容,请关注编程网行业资讯频道,感谢大家的支持。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

C++中怎么手动创建一个内存池

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

C++中怎么手动创建一个内存池

今天就跟大家聊聊有关C++中怎么手动创建一个内存池,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。引言使用new expression为类的多个实例分配动态内存时,cookie导致内存
2023-06-20

C#中怎么动态创建一个数组

这期内容当中小编将会给大家带来有关C#中怎么动态创建一个数组,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。C#动态创建数组实例:ArrayList al = new ArrayList(); al.A
2023-06-17

怎么在java中创建一个内存布局

本篇文章给大家分享的是有关怎么在java中创建一个内存布局,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。Java可以用来干什么Java主要应用于:1. web开发;2. And
2023-06-14

C#中怎么创建一个动态图像按钮

本篇文章为大家展示了C#中怎么创建一个动态图像按钮,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。我们要创建的控件简单说就是一个动态渐变的C#动态图像按钮这个button控件依旧继承自UserCont
2023-06-18

c++怎么创建共享内存

在C++中,可以使用操作系统的系统调用或者第三方库来创建共享内存。1. 使用操作系统的系统调用:在Unix/Linux操作系统中,可以使用shmget函数来创建共享内存。具体步骤如下:```cpp#include #include #inc
2023-08-20

C#中怎么创建一个表单

这期内容当中小编将会给大家带来有关C#中怎么创建一个表单,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。C#创建表单这非常简单,只需要创建一个从 System.WinForms.Form 类中衍生出来的类,
2023-06-17

C#中怎么创建一个Access表

本篇文章给大家分享的是有关C#中怎么创建一个Access表,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。C#操作Access主要知识点如下:using System.Data.
2023-06-18

c++怎么创建一个类

在C++中,可以通过以下步骤来创建一个类:1. 声明类:在头文件(.h或.hpp)中声明类的名称和成员函数。例如:```cppclass MyClass {// 成员变量private:int myInt;// 成员函数public:voi
2023-08-16

C#中怎么创建一个DataSet对象

C#中怎么创建一个DataSet对象,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。1. 启动 Microsoft Visual Studio .NET。在文件菜单上,单击新
2023-06-17

C#中怎么创建一个Excel文件

这篇文章给大家介绍C#中怎么创建一个Excel文件,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。C#创建Excel文件实现实例://创建一个excel application Excel.Application xl
2023-06-17

C#中怎么创建一个文件夹

本篇文章为大家展示了C#中怎么创建一个文件夹,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。C#创建文件夹在不同虚拟目录下的实现方法是用Server.MapPath("相对路径").相对路径根据不同情
2023-06-17

C#中怎么创建一个Word文档

C#中怎么创建一个Word文档,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。C#创建Word文档实例:using MSExcel = Microsoft.Office.Inte
2023-06-17

C#中怎么创建一个XML文档

本篇文章为大家展示了C#中怎么创建一个XML文档,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。1.创建一个XML文档:这一点和***种方法一样,在这里就不介绍了。2.通过"XmlDocument"类
2023-06-17

C#中怎么创建一个Service模板

C#中怎么创建一个Service模板,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。打开VS 2005,点击File->New->Project…,在C#创建工程向导中选择W
2023-06-17

怎么在java中使用ThreadPoolExecutor创建一个线程池

这篇文章给大家介绍怎么在java中使用ThreadPoolExecutor创建一个线程池,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Java可以用来干什么Java主要应用于:1. web开发;2. Android开发
2023-06-14

C++内存管理中简易内存池怎么实现

这篇文章主要介绍“C++内存管理中简易内存池怎么实现”,在日常操作中,相信很多人在C++内存管理中简易内存池怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C++内存管理中简易内存池怎么实现”的疑惑有所
2023-06-22

C#中怎么创建一个类库工程

这篇文章将为大家详细讲解有关C#中怎么创建一个类库工程,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。C#语言有很多值得学习的地方,这里我们主要介绍C#类库工程,包括介绍使用对象浏览器浏览两个
2023-06-17

C# 中怎么创建一个复合控件

本篇文章为大家展示了C# 中怎么创建一个复合控件,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。构建提供丰富的客户端接口的复杂Web控件经常需要把一些客户端JavaScript代码与控件的服务器端代码
2023-06-17

C#中怎么创建一个Web Service服务

本篇文章为大家展示了C#中怎么创建一个Web Service服务,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。为什么需要Web Service?以前,分布式的应用程序逻辑需要使用分布式的对象模型,通
2023-06-18

全C#中怎么创建一个XML文件

今天就跟大家聊聊有关全C#中怎么创建一个XML文件,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。C#创建XML文件的相关概念:这篇文章假定读者具备一定的XML基础,如果你刚接触XML
2023-06-17

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录