使用std::map和std::list存放数据,消耗内存比实际数据大得多

场景:项目中需要存储一个结构,如下程序段中TEST_DATA_STRU,结构占24B。但是使用代码中的std::list<DataListMap>类存储4000个DataListMap,每个DataListMap中有4个pairs,每个pair中的DataList中有6000个items时,消耗掉的内存几乎是我们存放TEST_DATA_STRU的2倍。

#include <iostream>
#include <map>
#include <list>
#include <vector> typedef struct TEST_DATA_STRU
{
int Data_A;
int Data_B;
int Data_C;
int Data_D;
}TEST_DATA_STRU; typedef std::list<TEST_DATA_STRU> DataList;
typedef std::map<int, DataList> DataListMap; int main(int argc, char **argv)
{
std::cout << "create a map" << std::endl;
DataListMap dataListMap;
// there are 100 pairs in the dataListMap
for(int i=0; i<100; ++i)
{
DataList dataList;
// there are 1000 items in a dataList
for(int j=0; j<1000; ++j)
{
TEST_DATA_STRU testStru = {i, j, i+j, i-j};
dataList.push_back(testStru);
}
dataListMap.insert(make_pair(i, dataList));
} std::cout << "data size of DataListMap: " << sizeof(TEST_DATA_STRU) * 1000 + sizeof(int) * 100 << std::endl; std::cout << "testing..." << std::endl;
std::list<DataListMap> mapList;
for(int i=0; i<4000; ++i)
{
mapList.push_back(dataListMap);
} // finally the memory of mapList is about double of the data we want to save
return 0;
}

最后通过分析,排除了内存泄露等情况后,将原因锁定在DataListMap类上。进一步分析后才找到原因:我们存放的结构占用24B,但是std::map和std::list中的指针就会占用24B以上,所以最终std::map和std::list自身所需的内存几乎和我们存储的数据一样大,甚至更大。

深入分析:std::list和std::map属于散列容器,容器的空间之间是通过指针来关联的,所以指针会占用一部分内存,当自身存放的数据较2*8(std::list,双向链表)差别不大时,会有很大的额外内存开销。为了避免此开销,可以使用线性容器,std::vector。

修改代码如下:使用std::vector取代std::list

#include <iostream>
#include <map>
#include <list>
#include <vector> typedef struct TEST_DATA_STRU
{
int Data_A;
int Data_B;
int Data_C;
int Data_D;
}TEST_DATA_STRU; typedef std::list<TEST_DATA_STRU> DataList;
typedef std::map<int, DataList> DataListMap; typedef std::vector<TEST_DATA_STRU> DataVec;
typedef std::map<int, DataVec> DataVecMap; int main(int argc, char **argv)
{
std::cout << "create a map" << std::endl;
//DataListMap dataListMap;
DataVecMap dataVecMap;
// there are 100 pairs in the dataListMap
for(int i=0; i<100; ++i)
{
//DataList dataList;
DataVec dataVec;
dataVec.reserve(1000);
// there are 1000 items in a dataList
for(int j=0; j<1000; ++j)
{
TEST_DATA_STRU testStru = {i, j, i+j, i-j};
//dataList.push_back(testStru);
dataVec.push_back(testStru);
}
//dataListMap.insert(make_pair(i, dataList));
dataVecMap.insert(make_pair(i, dataVec));
} std::cout << "data size of DataListMap: " << sizeof(TEST_DATA_STRU) * 1000 + sizeof(int) * 100 << std::endl; std::cout << "testing..." << std::endl;
//std::list<DataListMap> mapList;
std::list<DataVecMap> mapList;
for(int i=0; i<4000; ++i)
{
//mapList.push_back(dataListMap);
mapList.push_back(dataVecMap);
} // finally the memory of mapList is almost same with the data we want to save
return 0;
}

最终省去了额外的内存开销。

记于2016.04.14 22:28:24。

使用std::map和std::list存放数据,消耗内存比实际数据大得多的更多相关文章

  1. 有关std::map和std::vector的使用

    先说map吧. 最需要注意的就是:用下标访问map中的元素时,与使用下标访问vector的行为截然不同! 用下标访问不存在的元素时,将导致在map容器中添加一个新的元素,它的键即为该下标! 然而很多时 ...

  2. C++ std::map用法简介

    #include "map" //引入头文件 初始化: std::map <int, std::string> _map1; //初始化 //c++11中引入的,可以直 ...

  3. C++ | 使用const std::map,map::[]时遇到的一个bug

    原函数简化后如下: void fun(const map<int,vector<int>> &mp, int index) { for (auto tmp : mp[i ...

  4. 基于Python项目的Redis缓存消耗内存数据简单分析(附详细操作步骤)

    目录 1 准备工作 2 具体实施   1 准备工作 什么是Redis? Redis:一个高性能的key-value数据库.支持数据的持久化,可以将内存中的数据保存在磁盘中,重启的时候可以再次加载进行使 ...

  5. 【EF学习笔记05】----------操作内存中的数据

    SingleOrDefault实验 //SingleOrDefault实验 using (var db = new Entities()) { var classes = new Classes() ...

  6. js中数据、内存、变量的概念及三者之间的关系

    目录 数据.内存.变量的概念及三者之间的关系 什么是数据 数据的特点 什么是内存 栈内存 堆内存 JS引擎如何管理内存 什么是变量 变量是普通类型时 变量是引用类型时 数据.内存.变量的三者之间的关系 ...

  7. std::map用法

    STL是标准C++系统的一组模板类,使用STL模板类最大的好处就是在各种C++编译器上都通用.    在STL模板类中,用于线性数据存储管理的类主要有vector, list, map 等等.本文主要 ...

  8. C++ std::map::erase用法及其陷阱

    1.引入: STL的map中有一个erase方法用来从一个map中删除制定的节点 eg: map<string,string> mapTest; typedef map<string ...

  9. std::map的clear()没有用?

    昨天晚上,我徒弟跑过来讲,他的程序的内存占用居高不下,愿意是std::map的clear()没有效果.于是我让他用erase(begin,end); 试试也不行. 代码如下: void release ...

随机推荐

  1. jQuery生成二维码 jquery.qrcode.js

    https://github.com/jeromeetienne/jquery-qrcode 1.将jquery.qrcode.min.js和jquery添加到您的网页中 <script src ...

  2. 深入理解php内核

    目录 第一部分 基本原理 第一章 准备工作和背景知识 第一节 环境搭建 第二节 源码布局及阅读方法 第三节 常用代码 第四节 小结 第二章 用户代码的执行 第一节 PHP生命周期 第二节 从SAPI开 ...

  3. SQL - 常用的特殊查询

    sql 查询某字段不重复的全部记录: select * from table where fid in(Select min(fid) FROM table group by name) :

  4. linux -- 基于zookeeper搭建yarn的HA高可用集群

    linux -- 基于zookeeper搭建yarn的HA高可用集群 实现方式:配置yarn-site.xml配置文件 <configuration> <property> & ...

  5. MYSQL--三大范式

    MYSQL--三大范式 范式简介: 设计关系数据库时,遵从不同的规范要求,设计出合理的关系型数据库,这些不同的规范要求被称为不同的范式,各种范式呈递次规范,越高的范式数据库冗余越小.目前关系数据库有六 ...

  6. tp5消息队列使用

    composer 安装 扩展包 composer require topthink/think-queue 注意版本:默认安装5.1的扩展包 地址: https://packagist.org/pac ...

  7. FTP服务器搭建(Centos7)

    1.1.1 查看是否安装vsftp rpm -qa | grep vsftpd 1.1.2 如果没有安装: yum -y install vsftpd 1.2.3 vsftpd.conf 配置文件 匿 ...

  8. weblogic CVE-2017-10271修复教程

    1.简介 CVE-2017-10271是weblogic wls-wsat组件的一个xml反序列化漏洞,可造成远程命令执行.更详细分析可见参考链接,本文强调在进行参考链接修复中的一些细节. 2.影响版 ...

  9. Nginx隐藏版本号操作

    1.定位当前nginx所使用的配置文件 ps -ef |grep nginx 如果-c参数.则-c文件即为nginx当前所用配置文件,如上图中配置文件即为/usr/local/nginx/conf/n ...

  10. MyBatis Spring整合配置映射接口类与映射xml文件

    本文转自http://blog.csdn.net/zht666/article/details/38706083 Spring整合MyBatis使用到了mybatis-spring,在配置mybati ...