栈Stack和队列Queue

news2025/1/19 23:28:11

目录

一、栈

(1)用数组实现

(2)用单链表实现

(3)用标注尾结点的单链表实现

(4)用双向链表实现

2、栈的实际应用

(1)改变元素的序列

(2)将递归转化为循环(逆序打印链表)

(3)括号匹配

(4)逆波兰表达式求值

(5)出栈入栈次序匹配

(6)最小栈

二、队列

1、队列的使用

 2、队列的模拟实现

3、循环队列

4、双端队列Deque

 三、面试题

1、用栈实现队列

2、用队列实现栈


一、栈

栈是一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作

进行数据插入和删除操作的一端称为栈顶,另一端称为栈底

栈中的数据元素遵守后进先出的原则

  • 压栈/进栈/入栈:栈的插入操作
  • 出栈:栈的删除操作
方法
功能
Stack()构造一个空的栈
E push(E e)e入栈,并返回e
E pop()将栈顶元素出栈并返回
E peek()获取栈顶元素
int size()获取栈中有效元素个数
boolean empty()检测栈是否为空

1、栈的模拟实现

(1)用数组实现
(2)用单链表实现

  • 若使用尾插法,则入栈时间复杂度为O(n),出栈时间复杂度为O(n)
  • 若使用头插法,则入栈复杂度为O(1),出栈复杂度为O(1)
(3)用标注尾结点的单链表实现

  • 尾插法:入栈O(1),出栈O(n)(因为删除尾结点依旧要遍历到前一个结点,改变其next值)
  • 头插法:入栈O(1),出栈O(1)
(4)用双向链表实现

无论尾插还是头插时间复杂度都为O(1)

 LinkedList<Integer> stack=new LinkedList<>();
        stack.push(1);// addFirst(e);
        stack.push(2);
        stack.push(3);
        System.out.println(stack.peek());//拿到头结点3
2、栈的实际应用
(1)改变元素的序列
(2)将递归转化为循环(逆序打印链表)
 //1、递归方式
    void reverseprintList1(Node head){
        if(head!=null){
            printList(head.next);
            System.out.print(head.val + " ");
        }
    }
//2、循环方式
    void reverseprintList2(Node head){
        if(head==null){
            return;
        }
        Stack<Node> s = new Stack<>();
        // 将链表中的结点保存在栈中
        Node cur = head;
        while(cur!=null){
            s.push(cur);
            cur = cur.next;
        }
        // 将栈中的元素出栈
        while(!s.empty()){
            System.out.print(s.pop().val + " ");
        }
    }
(3)括号匹配

        给定一个只包含' ( '、' ) '、' { '、' } '、' [ '、' ] '的字符串s,判断括号是否匹配。匹配条件:左括号必须与相同类型的右括号闭合,不可交错排列

        匹配示例:(){}[]、({}[])、([{}])、()[{}]

public boolean isValid(String s) {
        Stack<Character> stack=new Stack<Character>();
        for (int i = 0; i < s.length(); i++) {
            char c=s.charAt(i);
            if (c=='(' || c=='[' || c=='{'){
                //c为左括号就添加进来等着对称匹配
                stack.push(c);
            }else {
                //c为右括号
                //此时栈中可能有左括号等着匹配,也可能为空(无法匹配)
                if (stack.empty()){
                    return false;
                }
                //不为空,则栈中存放的是左括号,那就判断c与栈顶元素是否匹配(因为两括号必须要对称匹配)
                char top=stack.peek();
                if (c==')' && top=='(' || c==']' && top=='[' || c=='}' && top=='{'){
                    //对称匹配了则弹出此左括号
                    stack.pop();
                }else {
                    //遇到一个右括号无法与栈顶左括号对称匹配,则无法匹配
                    return false;
                }
            }
        }
        //如果s遍历完发现栈中不为空,即栈中仍有残存左括号未进行匹配,不匹配
        if (!stack.empty()){
            return false;
        }
        //否则匹配
        return true;
    }
(4)逆波兰表达式求值

中缀表达式转化为后缀表达式技巧:

public int evalRPN(String[] tokens) {
        Stack<Integer> stack=new Stack<>();
        for (String s : tokens) {
            if (!isOperation(s)) {
                //压入数字字符
                stack.push(Integer.parseInt(s));
            }else {
                //运算符则弹出栈顶2个元素进行操作(先弹出的放运算符右边)
                int num2=stack.pop();
                int num1=stack.pop();
                switch (s){
                    case "+":
                        stack.push(num1+num2);
                        break;
                    case "-":
                        stack.push(num1-num2);
                        break;
                    case "*":
                        stack.push(num1*num2);
                        break;
                    case "/":
                        stack.push(num1/num2);
                        break;
                    
                }
            }
        }
        return stack.pop();
    }

    private boolean isOperation(String s) {
        if (s.equals("+") || s.equals("-") || s.equals("*") || s.equals("/")){
            //是运算符
            return true;
        }
        return false;
    }
(5)出栈入栈次序匹配

 public boolean IsPopOrder (int[] pushV, int[] popV) {
        Stack<Integer> stack=new Stack<>();
        int j=0;
        for (int i = 0; i < pushV.length; i++) {
            stack.push(pushV[i]);
            while (!stack.empty() && j<popV.length && stack.peek()==popV[j]){
                stack.pop();
                j++;
            }
        }
        return stack.empty();
    }
(6)最小栈
class MinStack {
    private Stack<Integer> stack;
    private Stack<Integer> minStack;//存储阶段性最小值
    private int minValue;

    public MinStack() {
        stack=new Stack<>();
        minStack=new Stack<>();
    }

    public void push(int val) {
        stack.push(val);
        if (minStack.empty()){
            minStack.push(val);
        }else {
            if (val<=minStack.peek()){
                minStack.push(val);
            }
        }
    }

    public void pop() {
        if (!stack.empty()){
            int ret=stack.pop();
            if (minStack.peek()==ret){
                minStack.pop();
            }
        }
    }

    public int top() {
        //获取stack栈顶元素
        if (stack.empty()){
            return -1;
        }
        return stack.peek();
    }

    public int getMin() {
        //获取minStack栈顶元素
        if (minStack.empty()){
            return -1;
        }
        return minStack.peek();
    }
}
  • :一种数据结构。是一种特殊的线性表,只允许在栈顶进行插入和删除操作,栈顶的数据元素遵守后进先出的原则
  • 虚拟机栈:JVM内存管理的一部分,用于管理函数调用的内存和回收。属于线程私有,确保每个线程的内存隔离和安全。
  • 栈帧:函数调用过程中的内存管理单元。包含局部变量表、操作栈等信息。每个方法在运行时JVM都会创建一个栈帧,并将其压入虚拟机栈中;当方法调用结束时对应的栈帧会从虚拟机栈中出栈,确保函数调用的顺利进行和结束后的资源释放

二、队列

队列:只允许在一端进行插入和删除数据操作,在另一端进行删除数据操作的特殊线性表

  • 进行插入操作的一端称为队尾,进行删除操作的一端称为队头
  • 队列遵守先进先出的原则
1、队列的使用

在Java中,Queue是个接口,底层是通过链表实现的

方法
功能
boolean offer(E e)入队列
E poll()出队列
peek()获取队头元素
int size()获取队列中有效元素个数
boolean isEmpty()检测队列是否为空

Queue是个接口,在实例化时必须实例化LInkedList的对象,因为LinkedList实现了Queue接口

        Queue<Integer> q = new LinkedList<>();
        // 从队尾入队列
        q.offer(1);
        q.offer(2);
        q.offer(3);
        q.offer(4);
        q.offer(5);
        System.out.println(q.size());//5
        System.out.println(q.peek()); // 获取队头元素 1
        q.poll();//弹出队头元素 1
        System.out.println(q.poll()); // 从队头出队列,并将删除的元素返回 2
        if(q.isEmpty()){
            System.out.println("队列空");
        }else{
            System.out.println(q.size());//3
        }
 2、队列的模拟实现

队列中既然可以存储元素,那底层肯定要有能够保存元素的空间。通过前面线性表的学习了解到常见的空间类型有2种:顺序结构和链式结构。那是用哪种结构实现比较好呢?

(1)双向链表实现

(2)数组实现

如果rear==elem.length-1之后发现数组前面还有位置还可以往前面插入元素就好了,这时候也就是我们的循环队列

3、循环队列

数组设计循环队列 

4、双端队列Deque

双端队列是指允许两端都可以进行入队和出队操作的队列

Deque是一个接口,使用时必须创建LinkedList对象

在实际工程中,使用Deque接口是比较多的,栈和队列均可以使用该接口

Deque<Integer>stack =newArrayDeque<>();//双端队列的线性实现
Deque<Integer>queue=newLinkedList<>();//双端队列的链式实现

 三、面试题

1、用栈实现队列

class MyQueue {
    Stack<Integer> inStack;
    Stack<Integer> outStack;

    public MyQueue() {
        inStack=new Stack<>();
        outStack=new Stack<>();
    }

    public void push(int x) {
        inStack.push(x);
    }

    public int pop() {
        if (empty()){
            return -1;//队列此时为空
        }
        if (outStack.empty()){
            while (!inStack.empty()){
                outStack.push(inStack.pop());
            }
        }
        return outStack.pop();
    }

    public int peek() {
        if (empty()){
            return -1;
        }
        if (outStack.empty()){
            while (!inStack.empty()){
                outStack.push(inStack.pop());
            }
        }
        return outStack.peek();
    }

    public boolean empty() {
        return inStack.empty() && outStack.empty();
    }
}
2、用队列实现栈

class MyStack {
    private Queue<Integer> qu1;
    private Queue<Integer> qu2;


    public MyStack() {
        qu1=new LinkedList<>();
        qu2=new LinkedList<>();
    }

    //入栈入到不为空的队列,都为空则入到qu1即可
    public void push(int x) {
        if (!qu1.isEmpty()){
            qu1.offer(x);
        }else if (!qu2.isEmpty()){
            qu2.offer(x);
        }else {
            qu1.offer(x);
        }
    }

    //出栈出不为空的队列size-1个。最后一个元素就是要出栈的元素
    public int pop() {
        if (empty()){
            return -1;//栈为空
        }
        if (!qu1.isEmpty()){
            int size=qu1.size();
            for (int i = 0; i < size - 1; i++) {
                qu2.offer(qu1.poll());
            }
            return qu1.poll();
        }else {
            int size=qu2.size();
            for (int i = 0; i < size - 1; i++) {
                qu1.offer(qu2.poll());
            }
            return qu2.poll();
        }
    }

    public int top() {
        int tmp=-1;
        if (empty()){
            return -1;//栈为空
        }
        if (!qu1.isEmpty()){
            int size=qu1.size();
            for (int i = 0; i < size; i++) {
                tmp=qu1.poll();
                qu2.offer(tmp);//出的最后一个元素存在tmp即为栈顶元素
            }
            return tmp;
        }else {
            int size=qu2.size();
            for (int i = 0; i < size; i++) {
                tmp=qu2.poll();
                qu1.offer(tmp);
            }
            return tmp;
        }
    }

    public boolean empty() {
        return qu2.isEmpty() && qu1.isEmpty();
    }
}

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

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

相关文章

Tailscale 自建 Derp 中转服务器

文章目录 为什么要建立 Derp 中转服务器&#xff1f;安装 Go 环境通过 Go 安装 Derp处理证书文件自签一个域名启动 DerpIPV6 的支持防止 Derp 被白嫖以上的操作命令合集自建 Headscale 添加 Derp参考 为什么要建立 Derp 中转服务器&#xff1f; Tailscale 使用的算法很有趣: 所…

RPC安全可靠的异常重试

当调用方调用服务提供方&#xff0c;由于网络抖动导致的请求失败&#xff0c;这个请求调用方希望执行成功。 调用方应该如何操作&#xff1f;catch异常再发起一次调用&#xff1f;显然不够优雅。这时可以考虑使用RPC框架的重试机制。 RPC框架的重试机制 RPC重试机制&#xff1…

AutoDL部署视觉大模型llama3.2-vision,从视频中寻找特定目标

注&#xff1a; windows11系统。示例为此项目&#xff1a;https://github.com/win4r/VideoFinder-Llama3.2-vision-Ollama 在当今的人工智能领域&#xff0c;深度学习模型的计算需求日益增长&#xff0c;特别是在处理复杂的视觉任务时&#xff0c;强大的算力往往是实现高效应用…

【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法

【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法 目录 文章目录 【大语言模型】ACL2024论文-16 基于地图制图的罗马尼亚自然语言推理语料库的新型课程学习方法目录摘要&#xff1a;研究背景&#xff1a;问题与挑战&#xff1a;如何解…

golang调用模组程序实现交互输入自动化,获取imei及iccid

应用场景&#xff1a;在openwrt下调用移远的测试程序&#xff0c;并实现输入自动话&#xff0c;获取imei rootOpenWrt:~# ql-api-test Test groups:0: ql_dsi1: ql_nw2: ql_sim3: ql_dev4: ql_voice5: ql_sms6: ql_adc7: ql_i2c8: …

【数据分享】2022年我国10米分辨率茶树种植分布栅格数据

小麦、玉米、水稻、茶树等各类农作物的种植分布数据在农业、环境、国土等很多专业都经常用到&#xff01; 本次给大家分享的是我国2022年10米分辨率茶树种植分布栅格数据&#xff01;数据格式为TIFF格式。数据坐标为GCS_WGS_1984。数据格式为TIFF格式。数据坐标为GCS_WGS_1984…

【弱监督视频异常检测】2024-ESWA-基于扩散的弱监督视频异常检测常态预训练

2024-ESWA-Diffusion-based normality pre-training for weakly supervised video anomaly detection 基于扩散的弱监督视频异常检测常态预训练摘要1. 引言2. 相关工作3. 方法论3.1. 使用扩散自动编码器进行常态学习3.2. 全局-局部特征编码器3.2.1 局部块3.2.2 全局块3.2.3 协同…

vue实现展示并下载后端返回的图片流

// 点击下载 downLoadCode() {const image new Image();image.setAttribute("crossOrigin", "anonymous");image.onload () > {const canvas document.createElement("canvas");canvas.width image.width;canvas.height image.height;c…

STL关联式容器之平衡二叉搜索树

平衡二叉搜索树 在STL关联式容器介绍-CSDN博客中对二叉搜索树做了简要的描述&#xff1b;但是因为没有对二叉搜索树对数的深度及插入后树的结构进行调整&#xff0c;二叉搜索树可能失去平衡&#xff0c;造成搜寻效率低落的情况。如下所示&#xff1a; 所谓树形平衡与否&#xf…

Django启用国际化支持(2)—实现界面内切换语言:activate()

文章目录 ⭐注意⭐1. 配置项目全局设置&#xff1a;启用国际化2. 编写视图函数3. 配置路由4. 界面演示5、扩展自动识别并切换到当前语言设置语言并保存到Session设置语言并保存到 Cookie ⭐注意⭐ 以下操作依赖于 Django 项目的国际化支持。如果你不清楚如何启用国际化功能&am…

Java基础——继承和多态

目录 一、继承 继承的定义&#xff1a; 继承的基本用法&#xff1a; 如何调用父类的方法&#xff1f; 二、多态 多态性的好处 多态中的强制类型转换&#xff1a; 包的命名规则——域名倒叙 一、继承 继承的定义&#xff1a; 继承是面向对象编程中的一种机制&#xff0c…

2024-11-17 -MATLAB三维绘图简单实例

1. x -1:0.05:1; y x; [X, Y] meshgrid(x, y); f (X, Y) (sin(pi * X) .* sin(pi * Y)) .^ 2.*sin(2.*X2.*Y); mesh(X, Y, f(X, Y)); % 调用函数f并传递X和Y xlabel(X-axis); ylabel(Y-axis); zlabel(Z-axis); title(Surface Plot of (sin(pi * X) .* sin(pi * Y)) .^ 2.*…

resnet50,clip,Faiss+Flask简易图文搜索服务

一、实现 文件夹目录结构&#xff1a; templates -----upload.html faiss_app.py 前端代码&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widt…

Flink监控checkpoint

Flink的web界面提供了一个选项卡来监控作业的检查点。这些统计信息在任务终止后也可用。有四个选项卡可以显示关于检查点的信息:概述(Overview)、历史(History)、摘要(Summary)和配置(Configuration)。下面依次来看这几个选项。 Overview Tab Overview选项卡列出了以…

如何用Excel批量提取文件夹内所有文件名?两种简单方法推荐

在日常办公中&#xff0c;我们有时需要将文件夹中的所有文件名整理在Excel表格中&#xff0c;方便管理和查阅。手动复制文件名既费时又易出错&#xff0c;因此本文将介绍两种利用Excel自动提取文件夹中所有文件名的方法&#xff0c;帮助你快速整理文件信息。 方法一&#xff1…

微信小程序-prettier 格式化

一.安装prettier插件 二.配置开发者工具的设置 配置如下代码在setting.json里&#xff1a; "editor.formatOnSave": true,"editor.defaultFormatter": "esbenp.prettier-vscode","prettier.documentSelectors": ["**/*.wxml"…

Debezium日常分享系列之:Debezium3版本Debezium connector for JDBC

Debezium日常分享系列之&#xff1a;Debezium3版本Debezium connector for JDBC 概述JDBC连接器的工作原理消费复杂的Debezium变更事件至少一次的传递多个任务数据和列类型映射主键处理删除模式幂等写入模式演化引用和大小写敏感性连接空闲超时数据类型映射部署Debezium JDBC连…

前端页面自适应等比例缩放 Flexible+rem方案

在移动互联网时代&#xff0c;随着智能手机和平板电脑的普及&#xff0c;前端开发者面临的一个重要挑战是如何让网页在不同尺寸和分辨率的设备上都能良好地显示。为了应对这一挑战&#xff0c;阿里巴巴的前端团队开发了 flexible.js&#xff0c;旨在提供一种简单有效的解决方案…

Argo workflow 拉取git 并使用pvc共享文件

文章目录 拉取 Git 仓库并读取文件使用 Kubernetes Persistent Volumes&#xff08;通过 volumeClaimTemplates&#xff09;以及任务之间如何共享数据 拉取 Git 仓库并读取文件 在 Argo Workflows 中&#xff0c;如果你想要一个任务拉取 Git 仓库中的文件&#xff0c;另一个任…

AWTK VSCode 实时预览插件端口冲突的解决办法

AWTK XML UI 预览插件&#xff1a;在 vscode 中实时预览 AWTK XML UI 文件&#xff0c;在 Copilot 的帮助下&#xff0c;可以大幅提高界面的开发效率。 主要特色&#xff1a; 真实的 UI 效果。可以设置主题&#xff0c;方便查看在不同主题下界面的效果。可以设置语言&#xf…