C++之栈和队列使用及模拟实现

目录

栈的使用

 队列的使用

栈的模拟实现 

队列的模拟实现

deuqe容器介绍


在C语言中我们已经学习了栈和队列的相关性质,今天我们主要来学习C++语法中栈和队列的相关概念。

栈的使用

在C++中栈是一种容器适配器,在其内部适配了其它的容器,其相关接口使用适配的容器的相关接口进行实现。什么是适配器呢?下面我们模拟实现的时候会讲述。

栈的主要接口有:push(),pop(),top(),size(),empty().

#include<iostream>
#include<stack>
using namespace std;int main()
{stack<int> s;
//1.往栈中入数据s.push(1);s.push(2);s.push(3);s.push(4);
//2.求栈顶的元素cout<<s.top()<<endl;
//3.删除栈顶的元素s.pop();
//4.判断栈顶是否为空while (!s.empty()){cout << s.top() << " ";s.pop();}cout << endl;
//5.求栈中元素的个数cout << s.size() << endl;return 0;
}

 队列的使用

队列和栈一样,也是一个容器适配器。

队列的主要接口有:push(),pop(),front(),back(),size(),empty().

int main()
{queue<int> q;
//向队列中入元素q.push(1);q.push(2);q.push(3);q.push(4);
//获取队列头的数据cout << q.front() << endl;
//获取队列尾的数据cout << q.back() << endl;
//判断队列是否为空,删除队列头部的数据while (!q.empty()){cout << q.front() << " ";q.pop();} cout << endl;
//获取队列的元素的个数cout << q.size() << endl;return 0;
}

栈的模拟实现 

上面我们提到了栈是一种容器适配器,下来我们详细讲解一下什么是容器适配器。 我们之前学习过,栈可以是数组栈,也可以是链式栈。所以按照这个思路而言,栈的数据结构中我们完全可以采用vector数组或者list链表作为底层的容器去进行数据的存储以及接口的封装。但是有没有一种方法可以使底层容器既可以是链表也可以是数组呢。这个问题其实就回到了我们之前学习的模板的概念,因为一个模板在实例化时可以被转化为多种类型。所以这个容器适配器本质上其实也是一种模板,不过这个模板一般在实例化时都会被实例化成stl中的容器类型,比如vector,list,string,deque,forward_list等等。stl库中一般采用deque这个双端队列容器,至于为什么,下面我们讲deque时会为大家讲解。

代码如下。

#pragma once
#pragma once
#include<deque>
using namespace std;
namespace yjd
{template<class T, class Container = deque<T>>class stack{public://pushvoid push(const T& x){_con.push_back(x);}//popvoid pop(){_con.pop_back();}//sizesize_t size()const{return _con.size();}//emptybool empty() const{return _con.empty();}//topconst T& top() const{return _con.back();}private:Container _con;};}

队列的模拟实现

#pragma once
#include<deque>
using namespace std;
namespace yjd
{template<class T, class Container = deque<T>>class queue{public://pushvoid push(const T& x){_con.push_back(x);}//popvoid pop(){_con.pop_front();}//sizesize_t size()const{return _con.size();}//emptybool empty() const{return _con.empty();}//frontconst T& front() const{return _con.front();}//backconst T& back() const{return _con.back();}private:Container _con;};}

总的来说,栈和队列的实现都是采用deque容器存储数据,并封装deque的接口成了stack和queue对应的的接口,为什么使用deque容器,下来为大家解释。 

deuqe容器介绍

上文说道,stack和queue的容器适配器中我们选择了deque这个容器进行适配,为什么要选择这个容器适配呢?list和vector一样也支持对应的接口,为什么不去使用list和vector这两个容器呢?先来回忆一下vector和list的优缺点。

vector优点:底层物理空间是连续的,所以支持下标随机访问,cpu高速缓存命中率高。高速缓存命中率就是,cpu在访问数据时所需的数据已经加载到高速缓存中的比率。cpu在访问数据时,会先将数据加载到高速缓存中然后再进行访问。因为vector底层是连续的,所以加载第一个元素数据时,会顺便将后面的元素数据全部加载到高速缓存中,所以它的高速缓存命中率高。

vector缺点:头插头删效率低下,往往需要挪动整个数组的数据,复杂度为O(N)。不能按需申请释放空间,扩容时往往会二倍扩容,往往会开出很大的一块空间,造成空间的浪费,释放时会释放整个数组的空所以bu。

list优点:支持任意位置的插入删除,复杂度为O(1),因为只需要更改指针的指向。按需申请释放空间,比如new一个结点,delete这个结点。

list缺点:物理空间不连续,所以不支持下标随机访问

再来讲讲deque的结构。图示如下。

deque在底层申请空间时,先申请buffer1这块空间, 假设每个buffer空间可以存储8个元素,当buffer1满了时,在进行尾插时,我们申请了buffer3这块空间,进行元素的插入,要进行头插时,我们申请了buffer2这块空间,进行头插。并且为了保证deque的连续,必须在buffer2的尾部插入,在buffer3的头部进行插入。

中控指针数组中存储的就是每个buffer的地址,且buffer1的地址要放在中间的位置,其它buffer的地址要根据对应buffer的位置进行填入,保证数据的顺序存储。指针数组一般采用vector进行存储。

那么deque作为适配器容器相比vector和list的优点是什么呢?

先抛开deque作为适配容器不谈,我们先来对比deque和vector和list这两个容器。通过deque的底层结构我们可以看出,deque在进行头插时并不像vector那样要移动整个数组的元素,从这一方面来看,deque优于vector容器。且deque也支持重载[]进行随机访问,具体的访问方式为,先判断当前下标是否在第一个buffer里,没有在就先减去第一个buffer中的元素个数,然后进行除和模操作确定这些元素具体的位置,因为每个buffer的空间其实并不是连续的,所以这个随机访问也并不像vector那样随机,随机访问这一点又优于list容器。

所以其实deque是融合了vector和list优点的一个容器,至于为什么没有替代vector和list,是因为deque虽然融合了vector和list的优点,但是并没有将vector和list的优点发挥到极致,比如头插时,又要开辟一大块的空间,而list只需要创建一个节点,更改指针指向。比如[]随机访问,又不像vector那样随机。所以纵使deque融合了vector和list的优点,vector和list在stl容器中的地位仍然是无法撼动的。

deque作为stack和queue的适配器容器较vector和list的优点又是什么呢?

较vector的优势:push_back元素时,空间不够进行扩容,并不会像vector那样扩容二倍,扩完容之后也不会拷贝数据,不会造成空间资源的浪费。

较list的优势:在底层deque的物理结构也是部分连续的,只要是连续的物理空间,那么cpu在访问数据时,cpu的命中效率一定是很高的,而list底层的物理结构不连续,所以cpu命中率不高。其次,不需要像list那样频繁地申请和释放空间,一次开一个buffer的空间,所以申请和释放空间的代价低。

以上便是本期的所有内容。本期的重点为stack和queue的相关接口以及容器适配器的概念,deque这个容器相关内容作为了解即可。

本期内容到此结束^_^ 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/3269136.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

【数据结构】——双链表的实现(赋源码)

双链表的概念和结构 双链表的全称叫做&#xff1a;带头双向循环链表 它的结构示意图如下 注意&#xff1a;这⾥的“带头”跟前⾯我们说的单链表的“头结点”是两个概念&#xff0c;实际前⾯的在单链表阶段称呼不严谨&#xff0c;但是为了读者们更好的理解就直接称为单链表的头…

【计算机毕设论文】基于SpringBoot成绩管理系统

&#x1f497;博主介绍&#xff1a;✌全平台粉丝5W,高级大厂开发程序员&#x1f603;&#xff0c;博客之星、掘金/知乎/华为云/阿里云等平台优质作者。 【源码获取】关注并且私信我 感兴趣的可以先收藏起来&#xff0c;同学门有不懂的毕设选题&#xff0c;项目以及论文编写等相…

学习测试13-车载测试

车的发展 1&#xff0c;动力 VCU 是实现整车控制决策的核心电子控制单元 汽车驱动控制:车都是有VCU发出&#xff0c;驱动控制&#xff0c;电池&#xff0c;电机都是执行器。比如: 汽车启动:启动&#xff0c;发车&#xff0c;VCU发送指令到齿轮这些&#xff0c;开始转动启动&a…

C语言程序设计15

程序设计15 问题15_1代码15_1结果15_1 问题15_2代码15_2结果15_2 问题15_3代码15_3结果15_3 问题15_1 在 m a i n main main 函数中将多次调用 f u n fun fun 函数&#xff0c;每调用一次&#xff0c;输出链表尾部结点中的数据&#xff0c;并释放该结点&#xff0c;使链表缩短…

Shell脚本学习教程(菜鸟从入门到精通)

前言 这本教程是写给那些在UNIX环境下发现必须写些Shell 脚本&#xff0c;以利于工作进行的计算机用户与软件开发人员。例如&#xff0c;你可能是正在念计算科学的学生&#xff0c;手上有学校给你的第一个UNIX系统账号&#xff0c;你想知道在UNIX下更多的东西&#xff0c;例如…

T-CNN——利用张量 CNN 增强缺陷检测

1. 摘要 缺陷检测是制造业中一个重要而具有挑战性的问题。本研究引入了张量卷积神经网络&#xff08;T-CNN&#xff09;&#xff0c;并在罗伯特-博世制造工厂生产的超声波传感器组件缺陷检测的实际应用中验证了其性能。与同类 CNN 模型相比&#xff0c;作者的量子启发 T-CNN 通…

飞凌嵌入式亮相第七届全国大学生嵌入式芯片与系统设计竞赛北部赛区决赛现场

7月20日&#xff0c;2024年第七届全国大学生嵌入式芯片与系统设计竞赛北部赛区决赛在保定大学科技园正式开赛。本次大赛由全国大学生嵌入式芯片与系统设计竞赛组委会、北部赛区执委会主办&#xff0c;保定国家大学科技园与北京邮电大学联合承办&#xff0c;飞凌嵌入式作为本土嵌…

chrome浏览器驱动(所有版本)

chrome浏览器驱动 114之前版本 https://chromedriver.storage.googleapis.com/index.html 125以后 125以后版本下载链接在此&#xff0c;只有后面status是绿色对勾的才可以下载&#xff0c;驱动大版本一致就可以使用&#xff0c;不需版本号一模一样&#xff1b;下载所需版本只…

谨防评论插件暴露服务器 IP

不少评论区插件支持邮件推送&#xff0c;当有新评论的时候会发送邮件&#xff0c;这样就能及时知道有评论了。例如我使用的 Twikoo 就支持邮件推送&#xff08;还有其他方式&#xff0c;这里不展开&#xff09;。 但是&#xff0c;这个会暴露真实的服务器 IP。为此&#xff0c…

与Zoom集成获取会议开始和结束事件

一、注册一个Zoom免费帐号&#xff08;需要在国外注册&#xff0c;国内不允许&#xff09; 二、进入Zoom应用市场创建一个应用 点击”发展”&#xff08;开发&#xff09;菜单&#xff0c;选择构建应用。 同意条款&#xff1a; 选择应用类型&#xff1a; 设置应用信息&#x…

【第四天】计算机网络知识 HTTP1.0,HTTP1.1与HTTP2.0的区别 HTTP3.0

HTTP1.0&#xff0c;HTTP1.1与HTTP2.0的区别 HTTP1.0 默认是短链接&#xff0c;可以强制开启长连接。HTTP1.1默认长连接。HTTP2.0采用多路复用。 HTTP1.0&#xff1a; 默认使用短链接&#xff0c;每次请求都需要建立一个TCP连接。它可以设置&#xff1a;Connection: keep-aliv…

Spring Boot 与 MongoDB 整合指南

MongoDB MongoDB 是一种基于文档的NoSQL数据库&#xff0c;以其高性能、高可用性和易扩展性而著称。它使用 BSON&#xff08;类似 JSON 的二进制格式&#xff09;来存储数据&#xff0c;提供了灵活的数据模型&#xff0c;使得开发者可以更轻松地存储和查询复杂的数据结构。将M…

夯实数字经济的“新基建”-基于大数据与区块链技术的新型基础设施

随着我国数据市场的蓬勃发展&#xff0c;构建契合数据特性、加速数据流通与价值释放的新型数据基础设施变得尤为关键。数字基础设施作为数字经济蓬勃发展的基石&#xff0c;其完善与否直接关系到数据能否有效存储、顺畅流通及高效利用&#xff0c;进而促进数据资源向数据资产的…

Python 教程(四):Python运算符合集

目录 专栏列表前言1. 算术运算符2. 比较运算符3. 逻辑运算符4. 位运算符5. 赋值运算符6. 成员运算符7. 身份运算符总结 在前三篇教程中&#xff0c;我们学习了 Python 的基本语法和数据结构以及字符串的特性。本篇教程&#xff0c;我们将深入探讨 Python 中的运算符合集。 专栏…

【docker】部署证书过期监控系统mouday/domain-admin

证书过期了再去部署证书容易被骂&#xff0c;就找了一个开源的证书过期系统来部署一下 过程 官方文档&#xff1a;https://domain-admin.readthedocs.io/zh-cn/latest/manual/install.html#docker 直接下载镜像是超时的&#xff0c;切换一下文档推荐的镜像源 新建docker配置…

ERROR: Cannot find command ‘git’- do you have ‘git’ installed and in your PATH?

ERROR: Cannot find command ‘git’- do you have ‘git’ installed and in your PATH? 目录 ERROR: Cannot find command ‘git’- do you have ‘git’ installed and in your PATH? 【常见模块错误】 【解决方案】 欢迎来到英杰社区https://bbs.csdn.net/topics/61780…

Linux安装TrueNAS(网络附加存储)教程 –第1部分

TrueNAS CORE&#xff08;原名FreeNAS&#xff09;是一款流行的存储系统&#xff0c;可帮助您构建自己的高质量存储设置&#xff0c;而无需支付软件费用。您可以将其安装在计算机硬件或虚拟机 (VM) 上&#xff0c;以获得开源存储的好处。 您可以在家中、办公室或数据中心使用T…

vue element-ui日期控件传参

前端&#xff1a;Vue element-ui <el-form-item label"过期时间" :rules"[ { required: true, message: 请选择过期时间, trigger: blur }]"><el-date-picker v-model"form.expireTime" type"date" format"yyyy-MM-dd&…

LoRaWAN网络中的chirpstack

目录 一、chirpstack介绍 二、网关与chirpstack之间的通信 三、NS与AS之间的通信 1、Protobuf 2、gRPC 一、chirpstack介绍 ChirpStack 是一个开源的 LoRaWAN 网络服务器&#xff0c;可用于 设置私有或公共 LoRaWAN 网络。ChirpStack 提供了一个 Web 界面 用于管理网关、设…

重塑与整合奖励机制以对齐大模型

人工智能咨询培训老师叶梓 转载标明出处 大模型的对齐问题&#xff0c;即如何使模型的输出倾向于具备期望的属性&#xff08;如有帮助、无害、真实或创造性&#xff09;&#xff0c;是当前人工智能领域的热点问题。来自芝加哥大学、Google Research、Google DeepMind 和斯坦福大…