数据结构复盘——第三章:栈和队列

news2025/1/11 7:38:01

文章目录

    • 第一部分:栈
      • 1、栈的定义
      • 2、栈的操作
    • 第一部分习题
    • 第二部分:共享栈
      • 1、共享栈的定义
      • 2、共享栈的操作
    • 第二部分习题
    • 第三部分:链栈
      • 1、链栈的定义
      • 2、链栈的操作
    • 第三部分习题
    • 第一到三部分小结
      • 1、顺序栈和链栈的比较
      • 2、栈的应用
      • 3、栈的应用相关习题
    • 第四部分:队列
      • 1、队列的定义
      • 2、循环队列
      • 3、循环队列的操作
    • 第四部分习题
    • 第五部分:双端队列
      • 1、双端队列的定义
      • 2、双端队列的操作
    • 第六部分:链队
      • 1、链队的定义
      • 2、链队的操作
    • 第六部分习题
    • 第四到六部分小结
      • 1、循环队列与链式队列的比较
      • 2、队列的应用
    • 栈和队列小结


第一部分:栈

1、栈的定义

只允许在一端进行插入或删除操作的线性表,四个字:“后进先出”
向栈中插入元素称为入栈,从栈中删除元素称为出栈
出入栈示意图

2、栈的操作

空栈状态入栈操作满栈状态出栈操作
top = -1① top = top+1;
② A[top] = a;
top = Maxsize-1① e = A[top];
② top = top-1;
  • “上溢”现象——当栈满时,再作进栈运算产生空间溢出的现象;
  • “下溢”现象——当栈空时,再作退栈运算产生空间溢出的现象。

第一部分习题

  1. 若已知一个栈的入栈序列是1,2,3…,n,其输出序列为p1, p2,p3…, pn,若 p1=n,则 pi 为(C)。
    A.i
    B.n=i
    C.n-i+1(代数法得p2是n-1、p3是n-2,则归纳法得n-i+1)
    D.不确定

  2. 元素a1、a2、a3、a4依次进入顺序栈,则下列不可能的退栈序列是(D)。
    A.a4,a3,a2,a1(全部入栈后依次出栈)
    B.a3,a2,a4,a1(先入栈 a1-a3,然后出栈 a3 和 a2,再入栈 a4,最后依次出栈)
    C.a3,a4,a2,a1(先入栈 a1-a3,然后出栈 a3 ,再入栈 a4,最后依次出栈)
    D.a3,a1,a4,a2(很明显要出栈 a2 后才能出栈 a1)

  3. 设有一顺序栈S,元素s1、s2、s3、s4、s5、s6依次入栈,如果6个元素出栈的顺序是s2、s3、s4、s6、s5、s1,则栈的容量至少应该是(3)。-填空题
    【分析】:栈内最多同时存储了s6、s5、s1


第二部分:共享栈

1、共享栈的定义

共享栈又叫做双栈
两个栈共同并辟一个存储空间,让一个栈的栈底为该空间的始端(下标0),另一栈的栈底为该空间的末端(下标Maxsize-1)。当元素进栈时,都从两端向中间延伸,这样能够使剩余的空间为任意一个栈所使用。
双栈的结构示意图

2、共享栈的操作

  • 栈空: top1 = -1; top2 = Maxsize;
  • 栈满;top1 = top2-1时(top1+1 = top2也可);
  • 栈1的操作:
    • 入栈:① top1 = top1+1(向栈1顶移动);② A[top1] = val;
    • 出栈:① e = A[top1]; ② top1 = top1-1(向栈1底移动);
  • 栈2的操作:
    • 入栈:① top2 = top2-1(向栈2顶移动);② A[top2] = val;
    • 出栈:① e = A[top2];② top2 = top2+1(向栈2底移动)

第二部分习题

  1. 为了减小栈溢出的可能性,可以让两个栈共享一片连续存储空间,两个栈的栈底分别设在这片空间的两端,这样只有当(A)时才可能产生上溢
    A.两个栈的栈顶在栈空间的某一位置相遇
    B.其中一个栈的栈顶到达栈空间的中心点
    C.两个栈的栈顶同时到达栈空间的中心点(不可能出现的情况,或者说此时已经处于上溢)
    D.两个栈均不空,且一个栈的栈顶到达另一个栈的栈底(因为另一个栈不为空,所以最多到达栈底的前一个位置,和C中类似)

  2. 若栈采用顺序存储方式存储,现两栈共享空间V[1,…,m],top[1]、top[2]分别代表第1和第2个栈的栈顶,栈1的底在V[1],栈2的底在V[m],则栈满的条件是(B)。
    A. ∣ | top[2]-top[1] ∣ | = 0
    B.top[1]+1 = top[2](共享栈栈满的定义)
    C.top[1]+top[2] = m
    D.top[1] = top[2]


第三部分:链栈

1、链栈的定义

前面提到的栈和共享栈都是采用顺序存储的方式实现,如果是采用链式存储的方式实现的栈,我们称其为链栈。
其实链表实现的栈是对链表施加了一个“后进先出”约束,其结构和一般的链表是完全相同的。
链栈结构图

2、链栈的操作

需要注意,在前面顺序栈中top指的是数组下标,而链栈中top指的是指针。

  • 栈空:top = null;
  • 栈满:不存在滴!
  • 入栈:(其实就是链表的头插法)
    ① p->next = top;
    ② top = p;
  • 出栈:
    ① e = top->data;(p = top;)
    ② top = top->next;(free(p);)

第三部分习题

  1. 向一个栈顶指针为top的链栈中插入一个x结点,则执行(C)。
    A.top->next=x
    B.x->next=top->next;top->next=x
    C.x->next=top;top=x
    D.x->next=top;top=top->next

  2. 如果以链表作为栈的存储结构,则退栈操作时(D) 。
    A.必须判别栈是否满(链栈不存在栈满,除非是物理存储空间严重不足;且栈满是在入栈操作时判断)
    B.对栈不作任何判别(不做判断健壮性不足)
    C.判别栈元素的类型(没有意义)
    D.必须判别栈是否空


第一到三部分小结

1、顺序栈和链栈的比较

(1)时间性能比较
顺序栈和链栈的基本操作的算法,时间复杂度均为O(1)
(2)空间性能比较
初始时顺序栈必须确定一个固定的长度,所以有存储元素个数的限制和空间浪费的问题。
链栈无栈满问题,只有当内存没有可用空间时才会出现栈满,但是每个元素都需要一个指针域,从而产生了结构性开销。

  • 一般结论:当栈在使用过程中元素个数变化较大时(即难以确定固定的长度),用链栈比较好;反之,应该采用顺序栈。
  • 经典例题:和顺序栈相比,链栈有一个比较明显的优势是(A)。
    A.通常不会出现栈满的情况
    B.通常不会出现栈空的情况
    C.插入操作更容易实现
    D.删除操作更容易实现

2、栈的应用

  • 栈的应用:主要在括号匹配、表达式求值、递归中前缀表达式后缀表达式
  • 经典例题:中缀表达式 A − ( B + C / D ) ∗ E A-(B+C/D)*E A(B+C/D)E的后缀形式是(D)。
    A. A B − C + D / E ∗ AB-C+D/E* ABC+D/E
    B. A B C + D / − E ∗ ABC+D/-E* ABC+D/E
    C. A B C D / E ∗ + − ABCD/E*+- ABCD/E+
    D. A B C D / + E ∗ − ABCD/+E*- ABCD/+E

【分析】中缀表达式就是我们常见的运算符号在式子中间的算术式,比较难的是如何将其转化为前缀表达式后缀表达式,即运算符号在式子的前端或后端的算术式:

加括号:按照运算符优先级将每一步运算都用括号括起来,原括号>乘除>加减,即上式变为: ( A − ( ( B + ( C / D ) ) ∗ E ) ) (A-((B+(C/D))*E)) (A((B+(C/D))E))
移符号:将运算符移到括号前或括号后,即继续变为: − ( A ∗ ( + ( B / ( C D ) ) E ) ) -(A*(+(B/(CD))E)) (A(+(B/(CD))E)) ( A ( ( B ( C D ) / ) + E ) ∗ ) − (A((B(CD)/)+E)*)- (A((B(CD)/)+E))
去括号:把括号去除得到前缀表达式: − A ∗ + B / C D E -A*+B/CDE A+B/CDE,或后缀表达式: A B C D / + E ∗ − ABCD/+E*- ABCD/+E

3、栈的应用相关习题

  1. 将一个递归算法改为对应的非递归算法时,通常需要使用(A
    A.栈(通常使用栈,也可以使用其他方法)
    B.队列
    C.循环队列
    D.优先队列

  2. 算术表达式 A + B ∗ C − D / E A+B*C-D/E A+BCD/E转为前缀表达式后为(D) 。
    A. − A + B ∗ C / D E -A+B*C/DE A+BC/DE
    B. − A + B ∗ C D / E -A+B*CD/E A+BCD/E
    C. − + ∗ A B C / D E -+*ABC/DE +ABC/DE
    D. − + A ∗ B C / D E -+A*BC/DE +ABC/DE
    【分析】① ( ( A + ( B ∗ C ) ) − ( D / E ) ) ((A+(B*C))-(D/E)) ((A+(BC))(D/E));② − ( + ( A ∗ ( B C ) ) / ( D E ) ) -(+(A*(BC))/(DE)) (+(A(BC))/(DE));③ − + A ∗ B C / D E -+A*BC/DE +ABC/DE

  3. 中缀表达式 A ∗ ( B + C ) / ( D − E + F ) A*(B+C)/(D-E+F) A(B+C)/(DE+F)的后缀表达式是(C) 。
    A. A ∗ B + C / D − E + F A*B+C/D-E+F AB+C/DE+F
    B. A B + C + D / E − F + AB+C+D/E-F+ AB+C+D/EF+
    C. A B C + ∗ D E − F + / ABC+*DE-F+/ ABC+DEF+/
    D. A B C D E F ∗ + / − + ABCDEF*+/-+ ABCDEF+/
    【分析】① ( ( A ∗ ( B + C ) ) / ( ( D − E ) + F ) ) ((A*(B+C))/((D-E)+F)) ((A(B+C))/((DE)+F));② ( ( A ( B C ) + ) ∗ ( ( D E ) − F ) + ) / ((A(BC)+)*((DE)-F)+)/ ((A(BC)+)((DE)F)+)/;③ A B C + ∗ D E − F + / ABC+*DE-F+/ ABC+DEF+/


第四部分:队列

1、队列的定义

只允许在表的一端进行插入,而在表的另一端进行删除。四个字:“先进先出”
向队列中插入元素称为入队,从队列中删除元素称为出队
队列结构示意图
【注意】简单队列可能存在假溢出假满的问题,即:当一个满队列的队头元素出队后,虽然队头指针前产生了空位,但入队必须从队尾进行,而此时队尾指针已经达到最末端,无法后移(判定为队满)导致无法进行入队操作。
假满情况示意图

2、循环队列

为了解决假溢出或假满这一问题,一般将顺序队列组成循环队列。循环队列就是将队列的队首和队尾连接在一起,当然这只是一种逻辑结构,本质上还是顺序存储,实现的方法就是当队尾指针达到队列的最大位置时,再前进就回到队列起始位置。

3、循环队列的操作

空队状态入队操作满队状态出队操作
front = rear① queue[rear] = A;
② rear = (rear+1)%Maxsize;
(rear+1)%Maxsize = front① e = queue[front];
② front = (front+1)%Maxsize;
  • 上述操作中,对Maxsize取余是为了能够在到达队尾时+1回到队首。

第四部分习题

  1. 假设以数组 A[m] 存放循环队列的元素,其头尾指针分别为 front 和 rear,则当前队列中的元素个数为(A) 。
    A.(rear-front+m)%m
    B.rear-front+1
    C.(front-rear+m)%m
    D.(rear-front)%m
    【分析】因为 rear 是不可能超过 front 的,所以二者之间的差距最大为 m;用二者之差加上 m,若差值为正,取余运算将消去所加的 m;若差值为负,取余运算不会生效,该式相当于 ( m + r e a r ) − f r o n t (m+rear)-front (m+rear)front

  2. 设栈 S 和队列 Q 的初始状态为空,元素 a、b、c、d、e、f 依次通过栈 S,一个元素出栈后即进队列 Q,若6个元素出队的序列是 a、c、f、e、d、b,则栈S的容量至少应该是(C)。
    A.6
    B.5
    C.4
    D.3
    【分析】栈内最多的时候存储了b、d、e、f

  3. 若用一个大小为6的数组来实现循环队列,且当前 rear 和 front 的值分别为0和3,当从队列中删除一个元素,再加入两个元素后,rear和front的值分别为(B)。
    A.1和5
    B.2和4
    C.4和2
    D.5和1


第五部分:双端队列

1、双端队列的定义

指允许两端都可以进行入队和出队操作的队列(两端不受限)

  • 输出受限的双端队列:允许在一端进行插入和删除,但在另一端只允许插入的双端队列称为输出受限的双端队列.
  • 输入受限的双端队列:允许在一端进行插入和删除,但在另一端只允许删除的双端队列称为输入受限的双端队列.

2、双端队列的操作

与普通队列相同,在不受限的情况下可以对两端进行入队、出队操作。

  • 要注意如果两端都进行了入队操作,会导致先入队的元素被包围在双端队列中间。此时从某一端出队,变成了出口端入队的部分“后进先出”,入口端入队的部分“先进先出”。(输出受限)
  • 同样的如果只从某一端入队,却从两端出队,就变成了入口端出队的部分“后进先出”,出口端出队的部分“先进先出”。(输入受限)

【经典例题】若以1234作为双端队列的输入序列,既不能由输入受限的双端队列得到,也不能由输出受限的双端队列得到的输出序列的是(B)。
A.4213
B.4231
C.4132
D.1234
解题过程


第六部分:链队

1、链队的定义

实际上是一个同时带有队头指针和队尾指针的单链表,相当于我们对链表施加了一个“先进先出”的约束。
链队的结构示意图

2、链队的操作

需要注意,在前面顺序存储的队列中,front 和 rear 指的是数组下标,而链队中二者都是指针。

  • 空队:front = null;rear = null;
  • 满队:不存在滴!
  • 入队:(其实就是链表的尾插法)
    ① rear->next = p;
    ② rear = rear->next;
  • 出队:
    ① e = front->data;(p = front;)
    ② front = front->next;(free(p);)

第六部分习题

  1. 若用单链表来表示队列,则应该选用(B)。
    A.带尾指针的非循环链表(回到队首实现出队操作,过于耗时×)
    B.带尾指针的循环链表(尾插入队,然后循环到队首出队√)
    C.带头指针的非循环链表(去到队尾实现入队操作,过于耗时×)
    D.带头指针的循环链表(同样需要一个一个结点移到队尾,除非是双循环链表×)

  2. 判定“带头结点的链队列为空”的条件是(A) 。
    A.Q.front == NULL(头指针用于出队,一般指向队首元素)
    B.Q.rear == NULL(尾指针用于入队,本来就指向空)
    D.Q.front == Q.rear(可能是含有一个结点)
    D.Q.front != Q.rear


第四到六部分小结

1、循环队列与链式队列的比较

(1)时间性能比较
循环队列与链式队列基本操作的算法都需要常数时间O(1)

  • 循环队列是事先申请好空间,使用期间不释放,
  • 链式队列是每次申请和释放结点也会存在一些时间开销,

如果入队出队频繁,则两者还是有细微差异——最好选择循环队列,可以节省申请和释放结点的时间开销。

(2)空间性能比较
循环队列必须有一个固定的长度(顺序存储),所以就有了存储元素个数和空间浪费的问题。

2、队列的应用

  • 队列的应用:主要在层次遍历中,另外队列应用于计算机系统中可解决主机与外部设备之间速度不匹配的问题和多用户引起的资源竞争问题。
  • 经典例题:在解决计算机主机与打印机之间速度不匹配问题时通常设置一个打印数据缓冲区,这样主机将要输出的数据依次写入该缓冲区,而打印机则从该缓冲区中取出数据打印。该缓冲区应该是一个(B)结构。
    A.堆栈
    B.队列
    C.数组
    D.线性表

栈和队列小结

  • 栈在数据结构中考察占比较大,队列则主要在操作系统中考察占比较大。
    栈和队列的对比

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

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

相关文章

正向代理流程

正向代理场景 已经拿下了一台服务器,该服务器有两个网卡,一个公网,一个私网未被攻击的服务器只有一个内网网卡,它不能和其他网段通信,只能在本网段通信,被叫做不出网主机攻击机需要直接连接未被攻击的服务器…

Shopee印尼站停止销售跨境商品:电商新规是否将改变印尼电商格局?

在全球电商市场竞争激烈的背景下,东南亚地区的印尼一直以其庞大的人口和不断增长的中产阶级而闻名。这个国家拥有着无限的潜力,吸引着各大电商平台争相进军。 然而,近期,一项关于电子商务的新法规引发了广泛的讨论和关注。据报道…

竞赛选题 深度学习YOLO安检管制物品识别与检测 - python opencv

文章目录 0 前言1 课题背景2 实现效果3 卷积神经网络4 Yolov55 模型训练6 实现效果7 最后 0 前言 🔥 优质竞赛项目系列,今天要分享的是 🚩 **基于深度学习YOLO安检管制误判识别与检测 ** 该项目较为新颖,适合作为竞赛课题方向&…

C++ 快速排序算法

1、快速排序 步骤&#xff1a; 确定分界点 通常选 Q[L]、Q[(LR)/2]、Q[R]调整范围 使左边的 <X 右边 > X 也就是 左右两边的指针&#xff0c;依次比较定义的分界点&#xff08;X&#xff09;根据比较大小调整顺序 依次递归处理左右两端 模板代码&#xff1a; // 数组…

ExoPlayer架构详解与源码分析(5)——MediaSource

系列文章目录 ExoPlayer架构详解与源码分析&#xff08;1&#xff09;——前言 ExoPlayer架构详解与源码分析&#xff08;2&#xff09;——Player ExoPlayer架构详解与源码分析&#xff08;3&#xff09;——Timeline ExoPlayer架构详解与源码分析&#xff08;4&#xff09;—…

计算机毕业设计--基于SSM+Vue的物流管理系统的设计与实现

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;Vue 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#xff1a;是 目录…

Postman接口测试: postman设置接口关联,实现参数化

postman设置接口关联 在实际的接口测试中&#xff0c;后一个接口经常需要用到前一个接口返回的结果&#xff0c; 从而让后一个接口能正常执行&#xff0c;这个过程的实现称为关联。 在postman中实现关联操作的步骤如下&#xff1a; 1、利用postman获取上一个接口指定的返回值…

ICMP协议(一)

一 ICMP 说明&#xff1a; 了解大致内容即可,如果不是搞数通的只需要有个概念即可 小林 coding ① 概念 重点&#xff1a; ping、traceroute、mtr 主要是利用 ICMP 或者 UDP 的特性特点&#xff1a; ICMP 是TCP/IP协议簇的一个子协议,属于网络层 [三层]协议作用&#xff…

嵌入式开发常见的问题解决方法总结

本文引自 https://mp.weixin.qq.com/s/IBDnlzl_nFykemPxp7rt5w 一、问题复现 稳定复现问题才能正确的对问题进行定位、解决以及验证。一般来说&#xff0c;越容易复现的问题越容易解决。 (1) 模拟复现条件 有的问题存在于特定的条件下&#xff0c;只需要模拟出现问题的条件即…

外卖大数据案例

一、环境要求 HadoopHiveSparkHBase 开发环境。 二、数据描述 meituan_waimai_meishi.csv 是某外卖平台的部分外卖 SPU&#xff08;Standard Product Unit &#xff0c; 标准产品单元&#xff09;数据&#xff0c;包含了外卖平台某地区一时间的外卖信息。具体字段说明如下&am…

文件内容相关

1.查看文件 cat /etc/passwd 2.编辑文件 echo "i like dog" > qun.txt 标准输出重定向 echo "i like best cat" >> qun.txt 标准输出追加重定向 cat >> qun.txt cat >>qun.txt<< ene vim编辑 进入编辑模式 i 光标所在…

在unity中给游戏物体一个标记

标记 方便识别&#xff01; 标签&#xff08;Tag&#xff09; 引擎内部会对物体的标签建立了索引。通过标签查找物体&#xff0c;要比通过名字查找物体快得多。标签最多只能有 32个。前几个是常用标签&#xff0c;具有特定含义&#xff0c;例如玩家( Player)、主摄摄像机 (Mai…

【RTOS学习】优先级 | Tick | 任务状态 | 空闲任务 | 任务调度

&#x1f431;作者&#xff1a;一只大喵咪1201 &#x1f431;专栏&#xff1a;《RTOS学习》 &#x1f525;格言&#xff1a;你只管努力&#xff0c;剩下的交给时间&#xff01; 优先级 | Tick | 任务状态 | 空闲任务 | 任务调度 &#x1f3c0;优先级⚽任务管理 &#x1f3c0;T…

PostMan使用csv/json进行数据参数化

创建csv文件 或者创建json文件 [{"name": "zhangsan","age": 18},{"name": "lisi","age": 20} ] 运行集合脚本的时候选择data文件 在请求接口中输入全局变量 {{user}}的方式进行传递 在Tests中要使用断言&…

C# Winform编程(4)多文档窗口(MDI)

多文档窗口&#xff08;MDI&#xff09; 添加菜单&#xff0c;IsMdiContainer设为True: From窗口添加菜单 Form1.cs using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using S…

华为鸿蒙系统安装第三方软件 - 注意事项

华为鸿蒙系统安装第三方软件 - 注意事项 前言关闭增强防护关闭应用检测发现恶意软件解除软件管控 前言 华为鸿蒙系统默认开启纯净模式&#xff0c;仅支持安装经过华为应用市场检测的应用&#xff0c;并禁止运行病毒和风险应用。但此功能是可以关闭的&#xff0c;下文介绍如何安…

Qtday01(qt简介、简单窗口组件)

今日任务 仿qq登录界面&#xff0c;QT实现 代码&#xff1a; 头文件&#xff1a; #ifndef MAINWINDOW_H #define MAINWINDOW_H#include <QMainWindow> #include <QLineEdit> #include <QLabel> #include <QPushButton> #include <QtDebug> #…

session认证

目录 前言 http协议的无状态性 session的工作原理 在express中使用session认证 在session中存数据 在session中取数据 清空session 结尾 前言 session是一种记录客户状态的机制&#xff0c;客户端浏览器法访问服务器的时候&#xff0c;服务器把客户端信息以某种形式记录…

基于闪电连接过程优化的BP神经网络(分类应用) - 附代码

基于闪电连接过程优化的BP神经网络&#xff08;分类应用&#xff09; - 附代码 文章目录 基于闪电连接过程优化的BP神经网络&#xff08;分类应用&#xff09; - 附代码1.鸢尾花iris数据介绍2.数据集整理3.闪电连接过程优化BP神经网络3.1 BP神经网络参数设置3.2 闪电连接过程算…

互联网Java工程师面试题·Java 总结篇·第四弹

目录 31、String s new String(“xyz”);创建了几个字符串对象&#xff1f; 32、接口是否可继承&#xff08;extends&#xff09;接口&#xff1f;抽象类是否可实现&#xff08;implements&#xff09;接口&#xff1f;抽象类是否可继承具体类&#xff08;concrete class&am…