【C++】STL各容器的实现,时间复杂度,适用情况分析
一.vector
1.概述
动态数组,在内存中具有连续的储存空间,在堆上分配内存,支持快速随机访问,在中间插入和删除慢,但在末尾插入和删除快
2.特点
1)拥有一段连续的内存空间,并且起始地址不变,因此能非常好的支持随机存取,但由于其内存空间是连续的,所以在中间插入和删除会造成内存块的拷贝,另外,当该数组的内存空间不够时,需要重新申请一块足够大的内存并进行内存拷贝,这些都大大的影响了vector的效率
2)对头部和中间的元素进行插入删除需要移动内存,如果元素是结构体或者类,那么移动时还会进行析构和构造操作,所以性能不高
3)对末尾的元素的操作最快,此时一般不需要移动内存,只有剩余内存不够时才需要
3.时间复杂度分析:
头部插入删除:O(N)
尾部插入删除:O(1)
中间插入删除:O(N)
查找:O(N)
4.优缺点及适用场景
优点:支持随机储存,查询效率高
缺点:在头部和中间插入删除元素效率低,需要移动内存
适用场景:适用于元素结构简单,变化小,并且频繁随机访问的场景
5.结论
vector常用来保存需要经常进行随机访问的内容,并且不需要对中间元素进行添加和删除操作
二.双向队列deque
1.概述
deque是double ended queue的缩写,是一个动态数组,可以向两端发展(双向开口的连续线性空间),因此无论在头部或者尾部安插元素都十分迅速,在中间按插元素则比较费时,因为必须移动其他元素
双端队列的元素被表示为一个分段数组,容器中的元素分段存放在一个个大小固定的数组中,此外,容器还需要维护一个存放这些数组首地址的索引数组,如下图所示


由于分段数组的大小是固定的,并且他们的首地址被连续存放在索引数组中,因此可以对其进行随机访问,但是效率比vector低很多
向两端加入新元素时,如果这一端的分段数组未满,则可以直接加入,如果这一端的分段数组已满,只需要创建新的分段数组,并把该分段数组的地址加入到索引数组中即可,这样就不需要对已有的元素进行移动,因此在双端队列的两端加入新的元素都具有较高的效率
当双端队列删除首尾元素时,也不需要移动,所以效率也比较高
双端队列中间插入元素时,需要将插入点到某一端之间的所有元素向容器的这一端移动,因此向中间插入元素的效率较低,而且往往插入的位置越靠近中间,效率越低,删除队列中元素时,情况也类似
2.特点
1)按照页或者块来分配存储器,每页包含固定数目的元素
2)支持随机储存
3)头尾删除插入元素效率高
3.时间复杂度分析
头部尾部插入删除:O(1)
中间插入删除:O(N)
查找:O(N)
4.优缺点及适用场景
优点:头部尾部插入删除快
缺点:不适合中间插入删除操作,占用内存多
适用场景:适用于既要频繁随机储存,又要关心两端数据的插入与删除操作,并且不需要频繁对中间元素进行插入删除的场景
三.双向链表list
1.概述
list由双向链表实现,元素存放在堆中,每个元素都是存放在一块内存中,它的内存空间可以是不连续的,通过指针来进行数据访问,这个特点使得它的随机储存变得非常的没有效率,但是由于链表的特点,它可以很有效率的支持任意地方的插入和删除操作
2.特点
1)没有预留空间的习惯,所以每分配一个元素都会从内存中分配,每释放一个元素都会释放其占用的内存
2)任何地方插入删除的效率都很高,不需要移动内存,也就不需要在移动过程中对每个元素进行构造和析构,常用来做随机插入和删除的容器
3)访问首尾元素最快
3.时间复杂度分析
任何位置的插入删除:O(1)(假设告诉了你插入删除的位置,不需要你线查找再删除)
头尾查询:O(1)
其他位置查询:O(N)
4.优缺点及适用场景
优点:内存不连续,动态操作,在任意位置插入和删除的效率都高
缺点:不支持随机访问,相比于vector占用内存多
适用场景:适用于经常进行插入和删除并且不经常随机访问的场景
关于vector,deque,list的总结:
vector
特点:快速的随机储存,快速的在最后插入删除元素
适用:需要高效随机储存,需要高效的在末尾插入删除元素,不需要高效的在其他地方插入和删除元素
deque
特点:比较快速的随机存储(比vector的慢),快速的在头尾插入删除元素
适用:需要随机储存,需要高效的在头尾插入删除元素的场景
list
特点:快速在任意位置插入删除元素,快速访问头尾元素
适用:需要大量插入删除操作,不关心随机储存的场景
deque可以看作是vector和list的折中方案
四.set
1.概述
set由红黑树实现,其内部元素依照其值自动排序,每个元素只出现一次,不允许重复(红黑树是平衡二叉树的一种)
2.特点
1)元素有序
2)无重复元素
3)插入删除操作的效率比序列容器高,因为对于关联容器来说,不需要做内存的拷贝和内存的移动
3.时间复杂度分析
增删改查近似:O(log N)
4.优缺点及适用场景
优点:使用平衡二叉树实现,便于元素查找,而且保持了元素的唯一性,支持自动排序
缺点:每次插入元素,都需要调整红黑树,效率有一定的影响
适用场景:适用与经常查找一个元素是否在某集群中并且不要排序的场景
五.multiset
multiset和set相同,只不过它允许重复元素,也就是说multiset可包括多个数值相同的元素。这里不再做过多介绍。
六.map
1.概述
map由红黑树实现,其元素都是键值对,每个元素的键是排序的准则,每个键只能出现一次,不允许重复
2.特点
1)元素为键值对形式,键和值可以是任意类型
2)因为key有序,所以可以通过二分对key进行快速查找
3)增加和删除结点对迭代器的影响很小,除了当前结点是迭代器指向的结点
4)对于迭代器来说,可以修改值,但是不能修改key
3.时间复杂度分析
增删改查基本是O(log N)
4.优缺点和适用场景
优点:适用平衡二叉树实现,便于元素查找,而且可以把值映射到另外一个值,可以创建字典
缺点:每次插入都需要调整红黑树,对效率存在一定的影响
适用场景:适用于需要储存一个字典,并要求方便的根据key找value的场景
七.multimap
multimap和map相同,但允许重复元素,也就是说multimap可包含多个键值(key)相同的元素。这里不再做过多介绍。
ps:stack,queue,prioroty_queue都属于容器配接器,是由容器按照特殊的逻辑实现的
小结:
1)如果需要高效的随机存取,不在乎插入和删除的效率,使用vector。
2)如果需要大量的插入和删除元素,不关心随机存取的效率,使用list。
3)如果需要随机存取,并且关心两端数据的插入和删除效率,使用deque。
4)如果打算存储数据字典,并且要求方便地根据key找到value,一对一的情况使用map,一对多的情况使用multimap。
5)如果打算查找一个元素是否存在于某集合中,唯一存在的情况使用set,不唯一存在的情况使用multiset。
【C++】STL各容器的实现,时间复杂度,适用情况分析的更多相关文章
- 详解C++ STL multiset 容器
详解C++ STL multiset 容器 本篇随笔简单介绍一下\(C++STL\)中\(multiset\)容器的使用方法及常见使用技巧. multiset容器的概念和性质 \(set\)在英文中的 ...
- STL关联容器
这里简单学习一下STL关联容器,主要是map.multimap.set.multiset以及unordered_map.前四个底层实现都是利用红黑树实现的,查找算法时间复杂度为\(O(log(n))\ ...
- 2.7 C++STL list容器详解
文章目录 2.7.1 引入 2.7.2代码示例 2.7.3代码运行结果 总结 2.7.1 引入 STL list 容器,又称双向链表容器,即该容器的底层是以双向链表的形式实现的.这意味着,list 容 ...
- C++ STL vector容器学习
STL(Standard Template Library)标准模板库是C++最重要的组成部分,它提供了一组表示容器.迭代器.函数对象和算法的模板.其中容器是存储类型相同的数据的结构(如vector, ...
- STL List容器
转载http://www.cnblogs.com/fangyukuan/archive/2010/09/21/1832364.html 各个容器有很多的相似性.先学好一个,其它的就好办了.先从基础开始 ...
- STL之容器适配器queue的实现框架
说明:本文仅供学习交流,转载请标明出处,欢迎转载! 上篇文章STL之容器适配器stack的实现框架已经介绍了STL是怎样借助基础容器实现一种经常使用的数据结构stack (栈),本文介绍下第二种STL ...
- STL的容器算法迭代器的设计理念
1) STL的容器通过类模板技术,实现数据类型和容器模型的分离. 2) STL的迭代器技术实现了遍历容器的统一方法:也为STL的算法提供了统一性. 3) STL的函数对象实现了自定义数据类型的算法运算 ...
- c++复习:STL之容器
1 STL的string 1 String概念 string是STL的字符串类型,通常用来表示字符串.而在使用string之前,字符串通常是用char*表示的.string与char*都可以用来表示字 ...
- stl之容器、迭代器、算法几者之间的关系
转自:https://blog.csdn.net/bobodem/article/details/49386131 stl包括容器.迭代器和算法: 容器 用于管理一些相关的数据类型.每种容器都有它的优 ...
- STL Queue 容器
STL Queue 容器 Queue简介 queue是队列容器,是一种“先进先出”的容器. queue是简单地装饰deque容器而成为另外的一种容器. # ...
随机推荐
- Kafka Network层解析,还是有人把它说清楚了
我们知道kafka是基于TCP连接的.其并没有像很多中间件使用netty作为TCP服务器.而是自己基于Java NIO写了一套. 几个重要类 先看下Kafka Client的网络层架构. 本文主要分析 ...
- VMware——安装CentOS
VMware——安装CentOS 摘要:本文主要记录了在VMware虚拟机里安装CentOS的步骤. 下载操作系统 可以从下面的镜像地址去下载各种版本的CentOS,此次安装使用的版本是7.2: ht ...
- SSO单点登录和CAS
一.单点登录流程 =====客户端====== 1.拦截客户端的请求判断是否有局部的session 2.1如果有局部的session,放行请求. 2.2如果没有局部session 2.2.1请求中有携 ...
- Mysql、Oracle、SQLServer等数据库参考文档免费分享下载
场景 MySQL 是最流行的关系型数据库管理系统,在 WEB 应用方面 MySQL 是最好的 RDBMS(Relational Database Management System:关系数据库管理系统 ...
- gsoap生成webservice调用客户端接口
1.下载gsoap2.8 2.运行 wsdl2h.exe -o XXX.h XXX.wsdl wsdl文件可以是本地文件,也可以是服务器的wsdl,比如http://192.168.0.122:333 ...
- 完整且易读的微信小程序的注册页面(包含倒计时验证码、获取用户信息)
目录 1.页面展示 2.wxml代码 3.wxss代码 4.js代码 1.页面展示 2.wxml代码 <!--pages/register/register.wxml--> <scr ...
- 【已采纳】新项目第一次怎么上传到github里面
言归正传,最近学习了怎么将新创建的本地代码上传到github上,这里简单的记录一下,我喜欢使用命令行,这里全用命令行来实现,不了解git命令的可以去了解下. 第一步:建立git仓库 cd到你的本地 ...
- 基于Docker实现MySQL主从复制
前言 MySQL的主从复制是实现应用的高性能,高可用的基础.对于数据库读操作较密集的应用,通过使数据库请求负载均衡分配到不同MySQL服务器,可有效减轻数据库压力.当遇到MySQL单点故障中,也能在短 ...
- linux (09) nginx反向代理,负载均衡
一.nginx域名分配 1.创建一个新的服务器 server server{ listen 80; server_name www.s15oumei.com; #定义访问域名 location / { ...
- 2018710101110-李华《面向对象程序设计(java)》第十二周学习总结
项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这个作业的要求在哪里 https://www.cnblogs.com/nwnu-daizh/p ...