【数据结构】栈和队列的深度探索,从实现到应用详解

在这里插入图片描述

💎所属专栏:数据结构与算法学习 

💎 欢迎大家互三:2的n次方_

🍁1. 栈的介绍

栈是一种后进先出的数据结构,栈中的元素只能从栈顶进行插入和删除操作,类似于叠盘子,最后放上去的盘子最先拿下来。

🍁2. 栈的基本操作

  • 压栈(Push):将一个元素压入栈顶。
  • 出栈(Pop):移除并返回栈顶元素。
  • 栈顶元素(Peek):返回栈顶元素但不移除。
  • 判空(IsEmpty):检查栈是否为空。
  • 栈的大小(Size):返回栈中的元素个数。

栈的定义方法也是和ArrayList一样的,然后就是使用对象去调用栈的方法

public class Text {public static void main(String[] args) {Stack<Integer> stack1 = new Stack<>();stack1.push(1);stack1.push(2);stack1.push(3);System.out.println(stack1.pop());System.out.println(stack1.peek());System.out.println(stack1.isEmpty());System.out.println(stack1.size());}
}

🍁3. 栈的实现

首先,栈是通过数组实现的,就像之前实现的顺序表一样

public class MyStack {public int[] elem;public int usedSize;public MyStack() {this.elem = new int[10];}
}

接下来实现一些栈的基本操作

🍁3.1 push()

当1 2 3 4依次入栈时,如下图

入栈其实很简单,只需要把元素放进去,接着usedSize++就可以了,但是学习数据结构我们的思维要严谨,如果栈满了怎么办,所以还需要处理栈满的情况,栈满之后就扩容,扩容也是和之前的顺序表一样的,判断是否栈满了也很简单,只需要判断数组的长度和usedSize是否相等就可以了

   public void push(int val) {if(isFull()) {this.elem = Arrays.copyOf(elem,2*elem.length);}elem[usedSize++] = val;}public boolean isFull() {return usedSize == elem.length;}

🍁3.2 pop()

遵循后进先出的原则,第一次出栈取到的元素是4

那怎么去实现这个效果呢

只需要定义一个val,把栈顶的元素取出来,赋值给val,进行返回,同时出栈之后usedSize的值要减1

之后还需要考虑栈为空的情况,如果栈为空,肯定是不能再进行出栈的操作了,此时就需要抛出一个异常

    public int pop() {if(isEmpty()){throw new EmptyStackException();}int val = elem[usedSize - 1];usedSize--;return val;}public boolean isEmpty() {return usedSize == 0;}

判断栈为空只需要判断usedSize是否为空

🍁3.3 peek()

peek()是获取栈顶元素,但是不删除,这个其实更简单,只需要把下标为usedSize的元素进行返回就可以了,也不需要usedSize--

同时,还是需要处理一下栈为空的情况

    public int peek() {if(usedSize == 0){throw new EmptyStackException();}return elem[usedSize - 1];}

接下来把之前写的方法测试一下:

public class Text {public static void main(String[] args) {MyStack stack = new MyStack();stack.push(1);stack.push(2);stack.push(3);System.out.print(stack.pop() + " ");System.out.print(stack.pop() + " ");System.out.print(stack.peek() + " ");}
}

🍁4. 栈的使用场景

🍁4.1 链表实现栈

问题:单链表是否可以实现栈?

单链表其实是可以实现栈的,用链表实现的栈叫做链式栈

入栈:如果采用尾插法入栈,入栈的时间复杂度为O(n),如果给出last节点为O(1),出栈的时间复杂度为O(n),因为需要遍历到末尾才能入栈

如果采用头插法,入栈和出栈的时间复杂度都是O(1)

既然单链表可以实现栈了,那么双向链表肯定也可以实现栈,不论是头插还是尾插,双链表实现的栈出栈和入栈时间复杂度都是O(1)。

而且会发现,LinkedList也定义了栈的一些基本操作,可以当作栈来使用

🍁4.2 将递归转化为循环

一个典型的例子就是逆序打印链表,我们都知道,正常情况下,单链表是不能逆序打印的,递归的调用就类似于栈,最外面的一层先被打印,也就是最末尾元素最先打印,还可以通过栈来模拟递归

    //递归方式private void printList1(MySingleList.ListNode head){if(head!=null){printList1(head.next);System.out.print(head.value + " ");}}//循环方式private void printList2(MySingleList.ListNode head){if(head == null) return;Stack<MySingleList.ListNode> s = new Stack<>();MySingleList.ListNode cur = head;//将节点保存在栈中while(cur!=null){s.push(cur);cur = cur.next;}//打印栈中节点while(!s.empty()){System.out.print(s.pop().value + " ");}}

🍁5. 队列的介绍

队列是一种先进先出的数据结构。队列中的元素只能从队尾插入,从队首移除,类似于排队买票,最先排队的人最先买到票。

 Java中的Queue是一个接口,Deque叫做双端队列

🍁6. 队列的基本操作

  1. 入队(offer):将一个元素插入队尾。
  2. 出队(poll):移除并返回队首元素。
  3. 队首元素(Peek):返回队首元素但不移除。
  4. 判空(IsEmpty):检查队列是否为空。
  5. 队列的大小(Size):返回队列中的元素个数。

由于Queue是一个接口,不能直接创建对象,所以这里通过LinkedList来创建对象 

public class Text {public static void main(String[] args) {//使用接口的实现类LinkedList创建对象Queue<Integer> q = new LinkedList<>();q.offer(1);q.offer(2);q.offer(3);//出队System.out.println(q.poll());//获取对头元素System.out.println(q.peek());}
}

🍁7. 队列的实现

🍁7.1 双向链表实现队列

双向链表的话,和栈一样,入队和出队的操作时间复杂度为O(n),因为队列是先进先出的原则,入队就采用尾插的方法,出队也就是头删的方法

public class MyQueue {//双向链表static class ListNode {public int val;ListNode pre;ListNode next;public ListNode(int val) {this.val = val;}}ListNode first = null;ListNode last = null;public int usedSize = 0;//尾插的方法进行入队public void offer(int val) {ListNode node = new ListNode(val);if (isEmpty()) {first = last = node;} else {last.next = node;node.pre = last;last = last.next;}usedSize++;}//头删的方法进行出队public int poll() {if (first == null) {throw new EmptyQueueException("队列为空");}int value = first.val;first = first.next;if(first!=null){first.pre = null;}usedSize--;return value;}//获取队头元素public int peek() {if (first == null) {throw new EmptyQueueException("队列为空");}return first.val;}public boolean isEmpty() {return usedSize == 0;}
}

🍁7.2 数组实现的循环队列

如果采用正常的数组来实现队列的话就会有以下的弊端,

 

这样出队之后,数组前面的空间就会空出来,造成空间的浪费,那如何把这些空间也利用起来呢

使用这样的循环结构,就可以解决这个问题,也就是循环队列

front和rear在同一个位置时,表示队列为空,那么队列满了也是这样的情况,此时怎么区分呢?

1.定义一个size专门判断

2.添加标记,定义一个boolean类型的flag,表示走过没有

3.空出一个空间,此时rear.next == front就表示队列已满 

此时还有一个问题:对于以上的例子,怎么解决下标的问题,例如从7下标是怎么到0下标的

 也就是下标最后再往后怎么表示:

公式:(数组下标 - 偏移量) % 数组长度 

7 ~ 0  可以通过 (7 + 1)% 8 来表示

还有一种情况:下标最前再往前

例如 2 下标往前走到 7 下标

公式:(数组下标 + 数组长度 - 偏移量) % 数组长度 

(2 + 9 - 4)% 9,加上数组长度也就是为了避免负数的情况

代码

测试用例

测试用例

测试结果

622. 设计循环队列

我们通过力扣上的这道题来实现一下:

class MyCircularQueue {public int front;public int rear;public int[] elem;public MyCircularQueue(int k) {elem = new int[k + 1];//由于有一个位置空出来了,所以要额外再多一个位置}public boolean enQueue(int value) {if (isFull()) {return false;}elem[rear] = value;rear = (rear + 1) % elem.length;return true;}public boolean deQueue() {if (isEmpty()) {return false;}front = (front + 1) % elem.length;return true;}public int Front() {if (isEmpty()) {return -1;} else {return elem[front];}}public int Rear() {if (isEmpty())return -1;int index = (rear == 0 )? elem.length - 1 : rear - 1;return elem[index];}public boolean isEmpty() {return rear == front;}public boolean isFull() {return (rear + 1) % elem.length == front;}
}

🍁8. 双端队列 

在Java中,Deque(双端队列)是一个接口,它扩展了Queue接口。Deque支持在两端插入和删除元素,提供了比Queue更丰富的操作集合。可以使用Deque作为栈(后进先出)、队列(先进先出)、或者两者兼有。

Java提供了几种Deque的实现,其中最常见的是ArrayDequeLinkedListArrayDeque是基于数组的双端队列,它在大多数操作中都提供了更好的性能。而LinkedList也实现了Deque接口,但由于其基于链表的结构,它在添加和删除元素时可能不如ArrayDeque高效。

public class DequeDemo {public static void main(String[] args) {//线性实现Deque<Integer> deque = new ArrayDeque<>();deque.add(1);deque.add(2);deque.add(3);for(int i : deque){System.out.println(i);}//链式实现Deque<Integer> ldeque = new LinkedList<>();ldeque.add(1);ldeque.add(1);ldeque.add(1);for(int i : ldeque){System.out.println(i);}}
}

在这里插入图片描述

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

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

相关文章

广州机房搬迁网络部署方案

新机房网络部署应包括核心模块、业务模块、光传输模块、安全模块、流量采集模块、路由模块、带外管理模块等&#xff0c;每个模块都根据业务需求规划成多个POD&#xff08;Point Of Delivery&#xff0c;基本物理设计单元&#xff09;。 核心模块部署主要实现各业务模块的高速互…

HighConcurrencyCommFramework c++通讯服务器框架 :目录,修改标题,配置,日志打印

目录规划 nginx 根目录下的三个文件 makefile :编译项目的入口&#xff0c;编译项目从这里开始 config.mk&#xff1a;也是个配置脚本用来增加变动的东西&#xff0c;应付可变 common.mk&#xff1a;最核心的编译脚本&#xff0c;每个子目录都要被编译.cpp程序 配置文件 配…

postman创建mock server

B站博主的说明&#xff1a;

《python语言程序设计》第6章2题(求一个整数各个数字的和)编写一个函数

求一个整数各个数字的和编写一个函数&#xff0c;计算一个整数各个数字的和&#xff0c; def sumDigits(n):a n // 100b n % 100 // 10c n % 100 % 10print(f"{n}数&#xff0c;分成个&#xff0c;十&#xff0c;百&#xff0c;{a}{b}{c}", a b c)sumDigits(23…

算法日记day 16(二叉树的广度优先遍历|反转、对称二叉树)

一、二叉树的层序遍历 题目&#xff1a; 给你二叉树的根节点 root &#xff0c;返回其节点值的 层序遍历 。 &#xff08;即逐层地&#xff0c;从左到右访问所有节点&#xff09;。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;[[3]…

SQUID - 形状条件下的基于分子片段的3D分子生成等变模型 评测

SQUID 是一个形状条件下基于片段的3D分子生成模型&#xff0c;给一个3D参考分子&#xff0c;SQUID 可以根据参考分子的形状&#xff0c;基于片段库&#xff0c;生成与参考分子形状非常相似的分子。 SQUID 模型来自于 ICLR 2023 文章&#xff08;2022年10月6日提交&#xff09;&…

vue+watermark-dom实现页面水印效果

前言 页面水印大家应该都不陌生&#xff0c;它可以用于验证数字媒体的来源和完整性&#xff0c;还可以用于版权保护和信息识别&#xff0c;这些信息可以在不影响媒体质量的情况下嵌入&#xff0c;‌并在需要时进行提取。‌本文将通过 vue 结合 watermark-dom 库&#xff0c;教大…

OpenCV 像素操作—证件照换底色详细原理 C++纯手写实现

文章目录 总体步骤1.RGB转HSV2.找出要换的底色3.取反&#xff0c;黑白颠倒4.将原图像的非背景部分复制到新背景上 完整代码1.C纯手写版2.官方API版本 总体步骤 1.RGB转HSV 为什么一定要转为HSV 颜色空间&#xff1f; 将图像从BGR颜色空间转换为HSV颜色空间是因为HSV颜色空间更…

nginx基本原理

进程模型 当nginx启动之后&#xff0c;会有一个master进程和多个worker进程。默认是一个worker进程。 master进程的作用&#xff1a;接收来自外界信号&#xff0c;向各worker进程发送信号&#xff0c;监控worker进程的运行状态&#xff0c;当worker进程在异常情况下退出后&am…

C#实现数据采集系统-实现功能介绍

系统介绍 我们这里主要使用C#( .Net 6)来实现一个数据采集系统&#xff0c;从0到1搭建数据采集系统&#xff0c;从系统分析&#xff0c;功能拆解&#xff0c;到一一实现 数据采集 数据采集是企业信息化和数字化转型过程中的关键环节&#xff0c;它涉及到从生产设备、传感器…

jupyter_contrib_nbextensions安装失败问题

目录 1.文件路径长度问题 2.jupyter不出现Nbextensions选项 1.文件路径长度问题 问题&#xff1a; could not create build\bdist.win-amd64\wheel\.\jupyter_contrib_nbextensions\nbextensions\contrib_nbextensions_help_item\contrib_nbextensions_help_item.yaml: No su…

【艺术向】【素描创作记录】《如何为你的红颜知己创作一幅画像(之二)》

写在前面 之前分析过类似的创作过程&#xff0c;见博客【艺术向】【素描创作记录】《如何为你的红颜知己创作一幅画像》 本人业余时间修习素描多年&#xff0c;在此撰文记录《如何为你的红颜知己创作一幅画像&#xff08;之二&#xff09;》&#xff0c;博得对方好感&#xff…

C++常见问题

一、C入门基础 1.1、函数重载 函数重载允许在同一作用域内定义多个同名函数&#xff0c;只要这个函数的参数列表&#xff08;即参数的数量&#xff0c;类型或者顺序不同&#xff09; 如何支持&#xff1a;程序经过编译后&#xff0c;编译器会对程序中的函数按一定规则进行重…

设计模式-Git-其他

目录 设计模式&#xff1f; 创建型模式 单例模式&#xff1f; 啥情况需要单例模式 实现单例模式的关键点&#xff1f; 常见的单例模式实现&#xff1f; 01、饿汉式如何实现单例&#xff1f; 02、懒汉式如何实现单例&#xff1f; 03、双重检查锁定如何实现单例&#xff…

封装MAVSDK为JAR包并导出给其它Android工程用完整示例

效果: 未解锁状态 已执行解锁指令 已执行起飞指令 飞行中 已执行降落指令 已执行返航指令 实现步骤: 1.准备PX4容器并启动:

ip地址是电脑还是网线决定的

在数字化时代的浪潮中&#xff0c;网络已经成为了我们日常生活和工作不可或缺的一部分。当我们谈论网络时&#xff0c;IP地址无疑是一个核心的概念。然而&#xff0c;关于IP地址的分配和决定因素&#xff0c;很多人可能存在误解。有些人认为IP地址是由电脑决定的&#xff0c;而…

springboot nacos的各种注解、手动操作监听配置变化(监听指定DataId/监听任何变化)

文章目录 springboot nacos监听配置变化&#xff08;监听指定DataId/监听任何变化&#xff09;监听任何配置变化Nacos注解NacosConfigurationPropertiesNacosValueNacosConfigListenerNacosInjectedNacosConfigServiceNacosNamingService springboot nacos监听配置变化&#xf…

QT--事件(丰富操作,高级功能)

一、事件 1.事件与信号的区别 事件来自外部&#xff0c;是随机发生的。信号来自内部&#xff0c;是主动发生的。有点像外中断和内中断的区别。事件&#xff1a;适用于处理系统级别的输入和状态变化&#xff0c;种类繁多&#xff0c;能够应对复杂的交互需求。信号/槽&#xff…

中国 X86 CPU 技术源自何方

注&#xff1a; 原文发布于 2017 年&#xff0c;两篇合二为一。未与作者沟通&#xff0c;侵权&#xff0c;立删。 导语&#xff1a; Intel 对 X86 的授权有着极为严格的限制&#xff0c;那么上海兆芯的 X86 芯片技术到底从何而来&#xff1f;ZX-C 目前的短板在哪里&#xff1f;…

pytorch 46 将ASpanFormer模型导出onnx运行

ASpanFormer是一个2022年8月份发布的算法,其主要步骤与LoFTR模型类似,因此无法导出为onnx模型。根据ASpanFormer论文中的数据与效果图,可以确定AsPanFormer是可以作为一个比SP+SG更为有效的方案,其在标准数据集上的效果优于SP+SG,在速度上远超SP+SG,与LoFTR接近;在预测点…