“探秘数据结构:栈的奇妙魔力“

news2025/1/10 17:34:16

每日一言

兰有秀兮菊有芳,怀佳人兮不能忘。 —刘彻-


栈的概念及结构

栈(Stack) :一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。类似于一个垂直摞起来的盘子,想要拿或放只能从顶上,而不能从底部操作。
在这里插入图片描述

栈的一些操作

压栈(push):栈的插入操作叫做进栈/压栈/入栈,入数据在栈顶。
出栈(pop):栈的删除操作叫做出栈。出数据也在栈顶
获取栈顶元素(top):获取栈顶元素,但不将其从栈中移除。
判断栈是否为空(isEmpty):判断栈是否为空操作会返回一个布尔值,表示栈是否为空。

在这里插入图片描述

栈的实现

顺序栈

顺序栈是一种基于数组实现的栈数据结构。
在这里插入图片描述

代码实现

//准备工作

	int[] array;
    int size;//当前栈内的空间大小

    //创建栈,为栈分配存储空间
    public MyStack() {
        array = new int[3];
    }

//入栈

public void push(int val) {
        //检查是否需要扩容
        ensureCapacity();

        //入栈,并将栈的大小自增1
        array[size++] = val;
    }

//出栈

public int pop() {
		//因为top()方法中已经检查了栈是否为空,所以这里就不再检查了
        int val = top();
        size--;
        return val;
    }

//判断栈是否为空

public boolean isEmpty() {
        return size == 0;
    }

//获取栈顶元素

public int top() {
        if (isEmpty()) {
            System.out.println("栈为空!");
            return -1;//这里也可以写成抛出一个异常
        }
        return array[size - 1];
    }

//检查扩容

public void ensureCapacity() {
		//检查栈是否已满
        if (size == array.length) {
        	//已满,将栈的容量扩大一倍
            array = Arrays.copyOf(array, size * 2);
        }
    }

链栈

采用链式存储结构实现的栈称为链栈,链栈通常采用单链表来实现,因此其结构与单链表的结构相同由于栈的插入和删除操作仅限制在栈顶位置进行,所以采用单链表的表头指针作为栈顶指针。

代码实现
//准备工作

	private Element base;//栈底指针
    private Element top;//栈顶指针
    
    //栈中的每个节点
    private class Element {
        private int data;
        private Element next;
    }

//入栈

public void push(int val) {
        Element newElem = new Element();
        newElem.data = val;
        newElem.next = top;
        top = newElem;
    }

//出栈

 public int pop() {
        if(isEmpty()) {
            System.out.println("栈为空!");
            return -1;//这里也可以抛出异常
        }
        int val = top.data;
        top = top.next;
        return val;
    }

//判断栈是否为空

public boolean isEmpty() {
        //栈底指针和栈顶指针指向同一位置,此时栈为空
        return base == top;
    }

//获取栈顶元素

public int top() {
        return top.data;
    }

栈小结

⭐栈的特点是?
先进后出

⭐顺序栈与链栈哪个更好?

顺序栈和链栈各有优缺点,没有绝对的“更好”。选择哪种实现方式取决于问题的要求和特点。

⭐顺序栈(数组实现)的优点是:

  1. 存储空间连续,可以利用数组的随机访问特性,对栈的操作具有较高的效率。
  2. 实现简单,代码量较少。

⭐顺序栈(数组实现)的缺点是:

  1. 存储空间固定,创建栈时必须指定大小,大小无法动态调整。
  2. 当栈满时需要进行扩容,需要重新分配更大的内存空间,并将原有数据复制到新的内存空间中。

⭐链栈的优点是:

  1. 存储空间可以动态分配,不受固定大小的限制。
  2. 插入和删除元素的操作只需要修改指针,不需要移动大量的数据,相对较快。

⭐链栈的缺点是:

  1. 需要额外的指针指向下一个节点,增加了存储空间的开销。
  2. 由于每个节点需要存储指针,导致链栈的存储密度较低,对内存的利用率较低。

有关栈的一些题

1. 有效的括号

题目链接: 有效的括号

题目

给定一个只包括 ‘(’,‘)’,‘{’,‘}’,‘[’,‘]’ 的字符串 s ,判断字符串是否有效。

有效字符串需满足:

左括号必须用相同类型的右括号闭合。
左括号必须以正确的顺序闭合。
每个右括号都有一个对应的相同类型的左括号。

  • 示例 1:
    输入:s = “()”
    输出:true

  • 示例 2:
    输入:s = “()[]{}”
    输出:true

  • 示例 3:
    输入:s = “(]”
    输出:false

提示:
1 <= s.length <= 104
s 仅由括号 ‘()[]{}’ 组成


思路

  1. 遍历字符串中的每个字符。
  2. 如果字符是一个开括号(即’(', ‘[’, ‘{’),它会被压入栈中。
  3. 如果字符是一个闭括号,则检查栈是否为空。如果栈为空,意味着没有相应的开括号,字符串无效。如果栈不为空,则检查栈的顶部是否包含相应的开括号。如果是,栈的顶部元素将被弹出。如果不是,意味着字符串无效。
  4. 在遍历字符串的所有字符之后,检查栈是否为空。如果栈为空,意味着所有的开括号已经匹配并从栈中弹出,字符串有效。如果栈不为空,意味着还有剩余的开括号,字符串无效。

代码

class Solution {
    public boolean isValid(String s) {
        Stack<Character> stack = new Stack<>();
        for(int i = 0; i < s.length(); i++) {
            char ch = s.charAt(i);
            if(ch == '(' || ch == '[' || ch == '{') {
                stack.add(ch);
            }else {
                if(stack.isEmpty()){
                    return false;
                }else { 
                    if (ch == ')' && stack.peek() == '(' ||
                        ch == ']' && stack.peek() == '[' ||
                        ch == '}' && stack.peek() == '{' ) {
                    stack.pop();
                    } else {
                        return false;
                    }
                }
            }
        }
        return stack.isEmpty();
    }
}

2. 逆波兰表达式求值

题目链接: 逆波兰表达式求值

题目

给你一个字符串数组 tokens ,表示一个根据 逆波兰表示法 表示的算术表达式。

请你计算该表达式。返回一个表示表达式值的整数。

注意:
有效的算符为 ‘+’、‘-’、‘*’ 和 ‘/’ 。
每个操作数(运算对象)都可以是一个整数或者另一个表达式。
两个整数之间的除法总是 向零截断 。
表达式中不含除零运算。
输入是一个根据逆波兰表示法表示的算术表达式。
答案及所有中间计算结果可以用 32 位 整数表示。

  • 示例 1:
    输入:tokens = [“2”,“1”,“+”,“3”,“*”]
    输出:9
    解释:该算式转化为常见的中缀算术表达式为:((2 + 1) * 3) = 9

  • 示例 2:
    输入:tokens = [“4”,“13”,“5”,“/”,“+”]
    输出:6
    解释:该算式转化为常见的中缀算术表达式为:(4 + (13 / 5)) = 6

  • 示例 3:
    输入:tokens = [“10”,“6”,“9”,“3”,“+”,“-11”,““,”/“,””,“17”,“+”,“5”,“+”]
    输出:22
    解释:该算式转化为常见的中缀算术表达式为:
    ((10 * (6 / ((9 + 3) * -11))) + 17) + 5
    = ((10 * (6 / (12 * -11))) + 17) + 5
    = ((10 * (6 / -132)) + 17) + 5
    = ((10 * 0) + 17) + 5
    = (0 + 17) + 5
    = 17 + 5
    = 22

提示:
1 <= tokens.length <= 104
tokens[i] 是一个算符(“+”、“-”、“*” 或 “/”),或是在范围 [-200, 200] 内的一个整数

思路

用循环遍历整个字符串数组,如果当前字符为数字,则入栈;如果当前字符为运算符,则将栈中的两个数字弹出,根据运算符参与相应的运算。最后栈当中会剩余一个最终结果,return返回这个结果。

代码

class Solution {
    public boolean isOperation(String tmp) {
    return tmp.equals("+") || tmp.equals("-") ||tmp.equals("*") ||tmp.equals("/");
    }
    public int evalRPN(String[] tokens) {
        Stack<Integer> stack = new Stack<>();
        for(int i = 0; i < tokens.length; i++) {
            String tmp = tokens[i];
            if(!isOperation(tmp)) {
                stack.add(Integer.valueOf(tmp));
            } else {
                int num2 = stack.pop();
                int num1 = stack.pop();
                switch (tmp) {
                    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();
    }
}

3. 栈的压入、弹出序列

题目链接:栈的压入、弹出序列

题目

描述
输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否可能为该栈的弹出顺序。假设压入栈的所有数字均不相等。例如序列1,2,3,4,5是某栈的压入顺序,序列4,5,3,2,1是该压栈序列对应的一个弹出序列,但4,3,5,1,2就不可能是该压栈序列的弹出序列。

  1. 0<=pushV.length == popV.length <=1000
  2. -1000<=pushV[i]<=1000
  3. pushV 的所有数字均不相同

示例1

输入:[1,2,3,4,5],[4,5,3,2,1]

返回值:true

说明:可以通过push(1)=>push(2)=>push(3)=>push(4)=>pop()=>push(5)=>pop()=>pop()=>pop()=>pop()
这样的顺序得到[4,5,3,2,1]这个序列,返回true

示例2

输入:[1,2,3,4,5],[4,3,5,1,2]

返回值:false

说明:
由于是[1,2,3,4,5]的压入顺序,[4,3,5,1,2]的弹出顺序,要求4,3,5必须在1,2前压入,且1,2不能弹出,但是这样压入的顺序,1又不能在2之前弹出,所以无法形成的,返回false

思路

  1. 定义一个整型变量j用于指示出栈数组popV的位置,初始化为0;定义一个Stack对象stack来模拟入栈过程。
  2. 使用循环遍历入栈数组pushV,
  3. 如果当前入栈元素不等于出栈数组popV中指针j所指向的元素,则将当前入栈元素压入栈stack中。
  4. 如果当前入栈元素等于出栈数组popV中指针j所指向的元素,则进行出栈操作:
  5. 将指针j向后移动一位,表示出栈数组中的下一个元素。
  6. 检查栈顶元素是否与popV[j]相等,如果相等,则继续出栈,直到栈为空或者栈顶元素与popV[j]不相等。
  7. 判断最终栈是否为空:遍历结束后,如果栈为空,则说明入栈和出栈顺序是匹配的,返回true;否则返回false。

代码

import java.util.*;

public class Solution {
    public boolean IsPopOrder(int[] pushV, int[] popV) {
        int j = 0;
        Stack<Integer> stack = new Stack<>();
        for(int i = 0; i < pushV.length; i++) {
            if(pushV[i] != popV[j]) {
                stack.push(pushV[i]);
            } else {
                j++;
                while(!stack.isEmpty() && j < popV.length && popV[j] == stack.peek()) {
                    j++;
                    stack.pop();
                }
            }
        }
        return stack.isEmpty();
    }
}

结语

栈的特点:先进后出

有两种实现栈的方法,我该用哪个?

如果存储空间的大小事先已知且固定,并且对栈的操作效率要求较高,可以选择顺序栈。如果存储空间的大小不确定,或者频繁地进行插入和删除操作,并且对栈的大小没有严格的限制,可以选择链栈。


都看到这里啦!真棒(*^▽^*)

可以给作者一个免费的赞赞吗,这将会鼓励我继续创作,谢谢大家

编程小白写作,如有纰漏或错误,欢迎指正


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

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

相关文章

软件资源分享六:EPLAN Electric P8 2024 | Eplan 2024 中文版软件介绍+保姆级安装教程

原文链接&#xff1a;安装激活教程 EPLAN Electric P8 2024 | Eplan 2024 中文版软件介绍安装教程 EPLAN 2024是一款电气设计软件&#xff0c;它可以用于自动化系统的设计、文档编制和维护。EPLAN可以对电气设计的各个方面进行完整的支持&#xff0c;包括电气控制系统、机械设…

全栈的自我修养 ———— redux入门(看这么一篇就够了!!!)

redux时react中负责状态管理的工具 一、下载二、配置1、目录2、store配置3、redux中index.js配置4、启动类中index.js配置 三、使用1、调用store的数据2、调用store里面的方法3、改变store里面的值 一、下载 npm I reduxjs/toolkit react-redux二、配置 1、目录 modules里面…

Pygame基础9-射击

简介 玩家用鼠标控制飞机&#xff08;白色方块&#xff09;移动&#xff0c;按下鼠标后&#xff0c;玩家所在位置出现子弹&#xff0c;子弹匀速向右飞行。 代码 没有什么新的东西&#xff0c;使用两个精灵类表示玩家和子弹。 有一个细节需要注意&#xff0c;当子弹飞出屏幕…

6、Cocos Creator 2D 渲染组件:​Sprite 组件​

Sprite 组件 Sprite&#xff08;精灵&#xff09;是 2D/3D 游戏最常见的显示图像的方式&#xff0c;在节点上添加 Sprite 组件&#xff0c;就可以在场景中显示项目资源中的图片。 属性功能说明Type渲染模式&#xff0c;包括普通&#xff08;Simple&#xff09;、九宫格&#x…

【云呐】公司资产怎么盘点,如何做好资产盘点?

固定资产盘点是一个复杂的过程&#xff0c;需要充分的计划、准备和组织。确保盘点团队具备足够的专业知识和技能&#xff0c;并与相关部门和人员进行充分的沟通和协作。  盘点公司的固定资产是确保资产准确性并管理资产风险的重要过程。下面是一般性的指导步骤&#xff0c;供…

Outlook邮箱后缀是什么?如何改邮箱后缀?

Outlook邮箱后缀可以更改吗&#xff1f;微软有哪些后缀的邮箱&#xff1f; 对于许多刚接触Outlook邮箱的新手来说&#xff0c;了解Outlook邮箱后缀是必不可少的一步。那么&#xff0c;Outlook邮箱后缀究竟是什么呢&#xff1f;接下来&#xff0c;AokSend就来详细探讨一下这个问…

element-ui badge 组件源码分享

今日简单分享 badge 组件的源码实现&#xff0c;主要从以下两个方面&#xff1a; 1、badge 组件页面结构 2、badge 组件属性 一、badge 组件页面结构 二、badge 组件属性 补充几个标签的用途&#xff1a; sub&#xff1a;下标、sup&#xff1a;上标、var 变量 代码如下&am…

YPay源支付V7开源版

YPay_V7版本即将停止维护更新&#xff0c;同时我们将开放最新版开源代码供学习和参考。虽然首批阶段的【function_8.1.php文件是加密的】&#xff0c;但授权已经除去&#xff0c;该代码将在新版YPay上线时开放给大家。我们也会定期进行迭代更新&#xff0c;随后将创建对应仓库&…

算法系列--动态规划--背包问题(2)--01背包拓展题目

&#x1f495;"2024.3.28小米汽车发布"&#x1f495; 作者&#xff1a;Lvzi 文章主要内容&#xff1a;算法系列–动态规划–背包问题(2)–01背包拓展题目 大家好,今天为大家带来的是算法系列--动态规划--背包问题(2)--01背包拓展题目 1.分割等和⼦集 链接: https:/…

风险与收益

风险与收益 影响资产需求的主要因素财富总量预期收益率资产的流动性影响流动性的主要因素 风险 如何降低风险系统风险和非系统风险机会集合与有效集合资产组合理论 影响资产需求的主要因素 影响资产需求的主要因素包括&#xff1a;财富总量、预期收益率、资产的流动性和风险。…

虚拟机与开发板之间互传文件、文件夹

1.配置桥接模式实现外网访问 1.1设置 VMnet0 要桥接的网卡 打开【编辑】-【虚拟网络编辑器】 选择【更改设置】 选择【VMnet0】&#xff0c;选择桥接到宿主机上的哪个网卡。 通过打开安装虚拟机的宿主机的【网络适配器】&#xff0c;可以查看网卡名称。 1.2虚拟机配置桥接模式…

支持多元AI场景应用,宁畅“NEX AI Lab”开放试用预约中

3月29日&#xff0c;宁畅在京举行发布会&#xff0c;正式发布“全局智算”战略&#xff0c;并在会上推出战略性新品“AI算力栈”&#xff0c;旨在有效解决大模型产业落地的全周期问题。 据宁畅CTO赵雷介绍&#xff0c;“AI算力栈”集成了宁畅在AI计算领域的软硬件能力&#xff…

Python中os.environ基本介绍及使用方法

文章目录 python中os.environos.environ简介os.environ进行环境变量的增删改查 python中os.environ的使用详解1. 简介2. key 字段详解2.1 常见 key 字段 3. os.environ.get() 用法4. 环境变量的增删改查和判断是否存在4.1 新增环境变量4.2 更新环境变量4.3 获取环境变量4.4 删除…

正大国际:安全合规的外盘期货途径

“外盘期货”一词是指在中国大陆以外建立的期货交易市场。交易所基于国内期货和外盘期货的全球定价、价格权威、巨大的外部交易量、成熟的交易市场和交易机制、强大的流动性、巨大的市场容量、在中国大陆没有控制和强劲的趋势。然而&#xff0c;许多人被引诱进入非法甚至非法平…

【MySQL】DML的表操作详解:添加数据&修改数据&删除数据(可cv例题语句)

前言 大家好吖&#xff0c;欢迎来到 YY 滴MySQL系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过C Linux的老铁 主要内容含&#xff1a; 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; YY的《C》专栏YY的《C11》专栏YY的…

【数字图像处理】图像的最近邻插值、双线性插值和双三次插值

图像最近邻插值、双线性插值和双三次插值 用 O ( X , Y ) O(X,Y) O(X,Y)表示 H W H\times W HW的原始图像&#xff0c; G ( X ^ , Y ^ ) G(\hat{X},\hat{Y}) G(X^,Y^)表示 H ^ Y ^ \hat{H}\times\hat{Y} H^Y^的目标图像。 最近邻插值 最近邻插值法令目标图像在 ( x ^ , y…

代码随想录-二叉树【从中序与后序遍历序列构造二叉树】

题目 根据一棵树的中序遍历与后序遍历构造二叉树。 注意: 你可以假设树中没有重复的元素。 例如&#xff0c;给出 中序遍历 inorder [9,3,15,20,7]后序遍历 postorder [9,15,7,20,3] 返回如下的二叉树&#xff1a; 思路 首先回忆一下如何根据两个顺序构造一个唯一的二叉…

JavaScript前端学习大全

一、概念 JavaScript简称为JS&#xff0c;这门语言诞生主要用于完成页面的数据验证&#xff0c;因此运行在客户端&#xff0c;需要浏览器来解析JavaScript的代码。是世界上最流行的脚本语言。JavaScript 是一种让网页变得有趣和动态的编程语言。比如&#xff0c;当你在网页上点…

Java的编程之旅44——学生信息管理系统

目录 1.MVC设计模式初探 文件结构的搭建 2.Student类用来初始化学生信息 3.主函数里的两个功能 1.调用初始化学生信息的功能 2.输出欢迎界面功能 4.Global类中方法的编写 5.StuPage类&#xff0c;StuCtrl类&#xff0c;StuModel类中方法的编写 1.查询功能 selStu方法的…

蓝桥杯真题:单词分析

import java.util.Scanner; //1:无需package //2: 类名必须Main, 不可修改 public class Main{public static void main(String[]args) {Scanner sannernew Scanner(System.in);String strsanner.nextLine();int []anew int [26];for(int i0;i<str.length();i) {a[str.charA…