STL vector, deque, list对比
2015-09-03 17:40
323 查看
内存使用对比(分析基于MS stl)
1. deque和vector的pop都不会释放内存,list的pop会释放内存
2. 但是当数量比较大的时候,deque的析构函数非常慢,vector的很快,list更慢
3. vector占用内存最小,deque其次,list最大
以vector为基准,deque和list多有一些功能方法
deque
pop_front
push_front
list
merge
pop_front
push_front
remove
remove_if
unique
选择条件
1. vector
向量 相当于一个数组, SGI版本使用malloc
在内存中分配一块连续的内存空间进行存储。支持不指定vector大小的存储。STL内部实现时,首先分配一个非常大的内存空间预备进行存储,即capacituy()函数返回的大小,当超过此分配的空间时再整体重新放分配一块内存存储,这给人以vector可以不指定vector即一个连续内存的大小的感觉。通常此默认的内存分配能完成大部分情况下的存储。
优点:(1) 不指定一块内存大小的数组的连续存储,即可以像数组一样操作,但可以对此数组
进行动态操作。通常体现在push_back() pop_back()
(2) 随机访问方便,即支持[ ]操作符和vector.at()
(3) 节省空间。
缺点:(1) 在内部进行插入删除操作效率低。
(2) 只能在vector的最后进行push和pop,不能在vector的头进行push和pop。
(3) 当动态添加的数据超过vector默认分配的大小时要进行整体的重新分配、拷贝与释
放
2. list
双向链表
每一个结点都包括一个信息快Info、一个前驱指针Pre、一个后驱指针Post。可以不分配必须的内存大小方便的进行添加和删除操作。使用的是非连续的内存空间进行存储。
优点:(1) 不使用连续内存完成动态操作。
(2) 在内部方便的进行插入和删除操作
(3) 可在两端进行push、pop
缺点:(1) 不能进行内部的随机访问,即不支持[ ]操作符和vector.at()
(2) 相对于verctor占用内存多
3. deque
双端队列 double-end queue
deque是在功能上合并了vector和list。
优点:(1) 随机访问方便,即支持[ ]操作符和vector.at()
(2) 在内部方便的进行插入和删除操作
(3) 可在两端进行push、pop
缺点:(1) 占用内存多
使用区别:
1 如果你需要高效的随即存取,而不在乎插入和删除的效率,使用vector
2 如果你需要大量的插入和删除,而不关心随即存取,则应使用list
3 如果你需要随即存取,而且关心两端数据的插入和删除,则应使用deque
内部实现
1. vector: 内部实现是数组,一段连续的内存。
2. list, 内部实现是双链表
3. deque 内部实现是内存块的链表。
4. string: 连续的内存
5. set,map: 红黑树(平衡二叉树的一种)
6. hash_map, hash_set 用哈希表(散列表)来实现。
7. stack: 用vector或者是deque来实现
8. queue,用deque实现
内存测试的代码
[cpp] view
plaincopy
// dequeTest.cpp : Defines the entry point for the console application.
//
#include "stdafx.h"
using namespace std;
void Test1()
{
deque<string> deq_str;
cout << sizeof(deq_str) << endl;
deq_str.push_back("s1");
cout << sizeof(deq_str) << endl;
deq_str.push_front("s2");
cout << sizeof(deq_str) << endl;
for(int i=0; i<100000; i++)
{
deq_str.push_back("sb");
}
deq_str.pop_front();
cout << sizeof(deq_str) << endl;
}
void Test2()
{
vector<string> vect_str;
for(int i=0; i<100000; i++)
{
vect_str.push_back("sb");
}
for(int i=0; i<100000; i++)
{
vect_str.pop_back();
}
vect_str.push_back("");
}
void Test3()
{
list<string> list_str;
for(int i=0; i<100000; i++)
{
list_str.push_back("sb");
}
for(int i=0; i<100000; i++)
{
list_str.pop_back();
}
list_str.push_back("sb");
}
int _tmain(int argc, _TCHAR* argv[])
{
Test1();
Test2();
Test3();
return 0;
}
1. deque和vector的pop都不会释放内存,list的pop会释放内存
2. 但是当数量比较大的时候,deque的析构函数非常慢,vector的很快,list更慢
3. vector占用内存最小,deque其次,list最大
以vector为基准,deque和list多有一些功能方法
deque
pop_front
push_front
list
merge
pop_front
push_front
remove
remove_if
unique
选择条件
1. vector
向量 相当于一个数组, SGI版本使用malloc
在内存中分配一块连续的内存空间进行存储。支持不指定vector大小的存储。STL内部实现时,首先分配一个非常大的内存空间预备进行存储,即capacituy()函数返回的大小,当超过此分配的空间时再整体重新放分配一块内存存储,这给人以vector可以不指定vector即一个连续内存的大小的感觉。通常此默认的内存分配能完成大部分情况下的存储。
优点:(1) 不指定一块内存大小的数组的连续存储,即可以像数组一样操作,但可以对此数组
进行动态操作。通常体现在push_back() pop_back()
(2) 随机访问方便,即支持[ ]操作符和vector.at()
(3) 节省空间。
缺点:(1) 在内部进行插入删除操作效率低。
(2) 只能在vector的最后进行push和pop,不能在vector的头进行push和pop。
(3) 当动态添加的数据超过vector默认分配的大小时要进行整体的重新分配、拷贝与释
放
2. list
双向链表
每一个结点都包括一个信息快Info、一个前驱指针Pre、一个后驱指针Post。可以不分配必须的内存大小方便的进行添加和删除操作。使用的是非连续的内存空间进行存储。
优点:(1) 不使用连续内存完成动态操作。
(2) 在内部方便的进行插入和删除操作
(3) 可在两端进行push、pop
缺点:(1) 不能进行内部的随机访问,即不支持[ ]操作符和vector.at()
(2) 相对于verctor占用内存多
3. deque
双端队列 double-end queue
deque是在功能上合并了vector和list。
优点:(1) 随机访问方便,即支持[ ]操作符和vector.at()
(2) 在内部方便的进行插入和删除操作
(3) 可在两端进行push、pop
缺点:(1) 占用内存多
使用区别:
1 如果你需要高效的随即存取,而不在乎插入和删除的效率,使用vector
2 如果你需要大量的插入和删除,而不关心随即存取,则应使用list
3 如果你需要随即存取,而且关心两端数据的插入和删除,则应使用deque
内部实现
1. vector: 内部实现是数组,一段连续的内存。
2. list, 内部实现是双链表
3. deque 内部实现是内存块的链表。
4. string: 连续的内存
5. set,map: 红黑树(平衡二叉树的一种)
6. hash_map, hash_set 用哈希表(散列表)来实现。
7. stack: 用vector或者是deque来实现
8. queue,用deque实现
内存测试的代码
[cpp] view
plaincopy
// dequeTest.cpp : Defines the entry point for the console application.
//
#include "stdafx.h"
using namespace std;
void Test1()
{
deque<string> deq_str;
cout << sizeof(deq_str) << endl;
deq_str.push_back("s1");
cout << sizeof(deq_str) << endl;
deq_str.push_front("s2");
cout << sizeof(deq_str) << endl;
for(int i=0; i<100000; i++)
{
deq_str.push_back("sb");
}
deq_str.pop_front();
cout << sizeof(deq_str) << endl;
}
void Test2()
{
vector<string> vect_str;
for(int i=0; i<100000; i++)
{
vect_str.push_back("sb");
}
for(int i=0; i<100000; i++)
{
vect_str.pop_back();
}
vect_str.push_back("");
}
void Test3()
{
list<string> list_str;
for(int i=0; i<100000; i++)
{
list_str.push_back("sb");
}
for(int i=0; i<100000; i++)
{
list_str.pop_back();
}
list_str.push_back("sb");
}
int _tmain(int argc, _TCHAR* argv[])
{
Test1();
Test2();
Test3();
return 0;
}
相关文章推荐
- 使用C++实现JNI接口需要注意的事项
- 关于指针的一些事情
- c++ primer 第五版 笔记前言
- share_ptr的几个注意点
- Lua中调用C++函数示例
- Lua教程(一):在C++中嵌入Lua脚本
- Lua教程(二):C++和Lua相互传递数据示例
- C++联合体转换成C#结构的实现方法
- C++编写简单的打靶游戏
- C++ 自定义控件的移植问题
- C++变位词问题分析
- C/C++数据对齐详细解析
- C++基于栈实现铁轨问题
- C++中引用的使用总结
- 使用Lua来扩展C++程序的方法
- C++中调用Lua函数实例
- Lua和C++的通信流程代码实例
- C与C++之间相互调用实例方法讲解
- C++ Custom Control控件向父窗体发送对应的消息
- C++中拷贝构造函数的应用详解