【Java数据结构】关于栈的操作出栈,压栈,中缀表达式,后缀表达式,逆波兰表达式详解

在这里插入图片描述

🔥个人主页:努力学编程’

🔥内容管理:java数据结构
请添加图片描述

上一篇文章我们讲过了java数据结构的链表,对于链表我们使用了它的一些基本操作,完成了扑克牌小游戏的操作,如果你感兴趣的话,点击超链接观看:【java数据结构】基于java提供的ArrayList实现的扑克牌游戏-(附源码~),今天带大家学习的是数据结构中另一个非常重要的知识-栈

目录

  • 1栈的一些基础知识
  • java代码实现一个栈
  • 对于栈功能的详解
    • 1.压栈:
    • 2.出栈pop:
    • 3.获取栈顶元素peek:
  • 逆波兰表达式求值:
    • 中缀表达式:
    • 后缀表达式:
  • 那么如何实现中缀表达式和后缀表达式的转换:
    • 解题思路:
    • 代码实现

1栈的一些基础知识

在开始前,请牢记这句话:栈是一种先进后出的数据结构。栈(stack)是限定仅在表的一端进行操作的数据结构,请联系我们前文所学的,设想一个单链表我们只能够对其链表的表尾结点进行操作,而操作也只能够进行插入一个新的结点与删除最末尾的这个结点两个操作,而这样强限制性的‘链表’,就是我们所说的栈。

栈在底层逻辑的实现上可分为链表栈和数组栈:

栈分为数组栈和链表栈,其区别是数组栈使用数组进行功能的模拟,实现较为快速和便利,而链表栈使用链表的思路去设计,实现较为麻烦,但是其稳定不易出错;在链表栈中又分为静态链表栈和动态链表栈,静态链表栈给定栈的空间大小,不允许超过存储超过给定数据大小的元素,而动态栈使用的是自动创建空间的方法进行创建,只要符合机器的硬件要求以及编译器的控制,其理论上是极大的。

以下是实现一个栈的具体操作过程:

1.选择数据结构:
栈可以使用数组(或列表)或链表来实现。数组实现的栈通常更简单,因为它们具有随机访问的能力,而链表实现的栈可能需要更多的指针操作。

2.定义栈的操作:
定义栈的基本操作,包括入栈(push)、出栈(pop)、获取栈顶元素(peek)等

3.确定栈的大小(如果有限制): 如果栈有大小限制,则需要确定栈的最大容量。如果栈的大小是动态的,则可以跳过此步骤。

4.实现栈的基本操作:
根据选择的数据结构,实现栈的基本操作。如果使用数组实现,通常使用数组的末尾作为栈的顶部,入栈操作将元素添加到数组末尾,出栈操作将从数组末尾移除元素。如果使用链表实现,需要定义节点类,并根据栈的操作更改节点的连接关系。

5.处理边界情况:
在实现栈的操作时,要考虑边界情况,如栈为空时尝试出栈或查看栈顶元素,或者栈已满时尝试入栈。

6.测试栈的实现:
编写测试代码,确保栈的实现能够正常工作,并处理各种情况,如入栈、出栈、查看栈顶元素、栈为空或栈已满等。

7.优化(如果需要)
根据实际需求和性能要求,可以对栈的实现进行优化,例如使用动态数组实现以处理栈大小动态变化的情况,或者使用链表实现以节省内存空间。

通过以上步骤,你可以实现一个简单而有效的栈数据结构。

java代码实现一个栈

import java.util.ArrayList;
import java.util.EmptyStackException;

public class Stack {
    private ArrayList<Integer> stack;
    private int maxSize;

    // 构造函数
    public Stack(int maxSize) {
        this.maxSize = maxSize;
        this.stack = new ArrayList<>();
    }

    // 判断栈是否为空
    public boolean isEmpty() {
        return stack.isEmpty();
    }

    // 判断栈是否已满
    public boolean isFull() {
        return stack.size() == maxSize;
    }

    // 入栈操作
    public void push(int item) {
        if (isFull()) {
            System.out.println("Stack is full. Cannot push element.");
        } else {
            stack.add(item);
        }
    }

    // 出栈操作
    public int pop() {
        if (isEmpty()) {
            throw new EmptyStackException();
        }
        return stack.remove(stack.size() - 1);
    }

    // 获取栈顶元素但不移除
    public int peek() {
        if (isEmpty()) {
            throw new EmptyStackException();
        }
        return stack.get(stack.size() - 1);
    }

    // 获取栈的大小
    public int size() {
        return stack.size();
    }

    // 主函数用于测试
    public static void main(String[] args) {
        Stack stack = new Stack(5);

        // 入栈操作
        stack.push(1);
        stack.push(2);
        stack.push(3);
        stack.push(4);
        stack.push(5);

        // 获取栈的大小
        System.out.println("Stack size: " + stack.size());

        // 出栈操作
        System.out.println("Pop element: " + stack.pop());
        System.out.println("Pop element: " + stack.pop());

        // 获取栈顶元素
        System.out.println("Peek element: " + stack.peek());

        // 获取栈的大小
        System.out.println("Stack size: " + stack.size());
    }
}

对于栈功能的详解

1.压栈:

向栈里插入数据,称为压栈,下面为图解:

在这里插入图片描述

既然栈我们底层用的是数组实现,那么在我们压栈的时候,就必须考虑一件事情,数组是否已满,如果数组已经放满了,我们就必须先对数组进行扩容,然后才能对数据进行插入

那么具体应该如何对栈实现扩容呢,其实和数组扩容是一个道理:

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

2.出栈pop:

将栈顶的元素弹出,后面的元素接替成为栈顶元素,属于栈的基本操作

在这里插入图片描述
下面是使用java实现的pop方法:

public int pop(){
        if(empty()){
            return -1;
        }
        int oldVal=elem[usedSize-1];
        usedSize--;
        return oldVal;
    }
    public boolean empty(){
        return usedSize==0;
    }

将栈顶的元素返回,具体对于栈数据的删除,其实就是将usedSize–,代表将数组的大小进行了调整。

3.获取栈顶元素peek:

本质上其实就是将栈顶元素进行打印,需要注意的一点这个方法返回栈顶的元素但不移除它。

 public int pop(){
        if(empty()){
            return -1;
        }
        int oldVal=elem[usedSize-1];
        usedSize--;
        return oldVal;
    }
    public boolean empty(){
        return usedSize==0;
    }
    public int peek(){
        if(empty()){
            return -1;
        }
        return elem[usedSize-1];
    }
}

以上就是对于栈的一些基本操作,了解完这些知识,我们就可以使用栈的知识解决一些算法题

逆波兰表达式求值:

点击链接一起挑战,逆波兰表达式求值要想解决这道题,我们需要首先了解中缀表达式,后缀表达式的知识

中缀表达式:

中缀表达式是我们最常见的数学表达式形式,它采用操作符位于操作数之间的形式。例如,“3 + 4”、“(5 - 2) * 6” 等都是中缀表达式。

中缀表达式的特点包括:

1.操作符在操作数中间:例如,“3 + 4” 中的加号就位于操作数 3 和 4 之间。
2.使用括号:中缀表达式通常需要使用括号来明确运算的优先级和顺序。
3.常见的运算符优先级规则:乘除法优先于加减法,同级运算符按照从左到右的顺序计算。

后缀表达式:

后缀表达式,也称为逆波兰表达式(Reverse Polish Notation,RPN),是一种不需要括号来表示运算顺序的数学表达式形式。在后缀表达式中,操作符在操作数之后出现,因此不需要括号来明确运算的顺序。

后缀表达式的主要特点是:

1.没有括号:
不需要括号来指定运算次序,因为操作符始终跟在操作数之后。
2.明确运算顺序:
从左到右扫描表达式,每当遇到一个操作符,就从栈中弹出足够的操作数进行计算,然后将结果压回栈中,直到整个表达式扫描完毕。
3.简化计算:
后缀表达式减少了运算符的优先级问题,简化了计算过程。
例如,将中缀表达式 "3 + 4 * 5" 转换为后缀表达式为 "3 4 5 * +"

处理后缀表达式的算法通常使用栈来实现。具体步骤如下:

1.从左到右遍历后缀表达式的每个字符。
2.如果遇到操作数,则将其压入栈中。
3.如果遇到操作符,则从栈中弹出相应数量的操作数进行运算,并将结果压入栈中。
4.最终栈中只剩下一个元素,即为后缀表达式的计算结果。

那么如何实现中缀表达式和后缀表达式的转换:

这里给大家分享一个方法
1.首先将表达式严格改为中缀表达式
2.把每个运算符转移到对应括号的后面
3.完成后缀表达式的转化
在这里插入图片描述

解题思路:

在这里插入图片描述

代码实现

 public int evalRPN(String[] tokens) {
        int n = tokens.length;
        int[] stack = new int[(n + 1) / 2];
        int index = -1;
        for (int i = 0; i < n; i++) {
            String token = tokens[i];
            switch (token) {
                case "+":
                    index--;
                    stack[index] += stack[index + 1];
                    break;
                case "-":
                    index--;
                    stack[index] -= stack[index + 1];
                    break;
                case "*":
                    index--;
                    stack[index] *= stack[index + 1];
                    break;
                case "/":
                    index--;
                    stack[index] /= stack[index + 1];
                    break;
                default:
                    index++;
                    stack[index] = Integer.parseInt(token);
            }
        }
        return stack[index];
    }

好了,今天就分享到这里,谢谢大家!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数组类模板(进阶版)

目录 介绍&#xff1a; 分析&#xff1a; 实现&#xff1a; .hpp框架创建 .hpp函数内容 有参构造 拷贝构造&#xff1a; 重载 插入数据 删除数据 通过下标访问 获取数组大小 获取数组容量 析构函数 .cpp框架 int类型数据测试 char类型测试 总代码 .hpp代码 …

是德科技keysight N9000B 信号分析仪

181/2461/8938产品概述&#xff1a; 工程的内涵就是将各种创意有机地联系起来&#xff0c;并解决遇到的问题。 CXA 信号分析仪具有出色的实际性能&#xff0c;它是一款出类拔萃、经济高效的基本信号表征工具。 它的功能十分强大&#xff0c;为一般用途和教育行业的用户执行测试…

wireshark 使用

wireshark介绍 wireshak可以抓取经过主机网卡的所有数据包&#xff08;包括虚拟机使用的虚拟网卡的数据包&#xff09;。 环境安装 安装wireshark: https://blog.csdn.net/Eoning/article/details/132141665 安装网络助手工具&#xff1a;https://soft.3dmgame.com/down/213…

【LIMS】CMA与CNAS:中国认证体系中的两大支柱

目录 一、CMA&#xff1a;[中国计量认证](http://cma-cma.org.cn/)什么是CMA&#xff1f;CMA的作用 二、CNAS&#xff1a;[中国合格评定国家认可委员会](https://www.cnas.org.cn/)什么是CNAS&#xff1f;CNAS的作用 三、CMA与CNAS的关系相互促进共同目标 结语系列文章版本记录…

TCP网络协议栈和Posix网络部分API总结

文章目录 Posix网络部分API综述TCP协议栈通信过程TCP三次握手和四次挥手&#xff08;看下图&#xff09;三次握手常见问题&#xff1f;为什么是三次握手而不是两次&#xff1f;三次握手和哪些函数有关&#xff1f;TCP的生命周期是从什么时候开始的&#xff1f; 四次挥手通信状态…

git基本操作二(小白快速上手)

1、前言 接上篇我们接着来继续讲 2、.gitignore忽略文件 创建一个.gitignore文件&#xff0c;并将其置于项目的根目录下&#xff0c;Git将自动识别并根据该规则忽略相应的文件和目录。 # 忽略所有的 .log 文件 *.log# 但跟踪所有的 build.log 文件 !build.log# 忽略所有的 /lo…

lookup函数

lookup函数 单条件查询 示例 扩展多条件 扩展

文件的顺序读写——顺序读写函数——fgets、fgetc、fputs、 fputc

✨✨ 欢迎大家来到莉莉的博文✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 目录 一、fgetc和fputc函数 1.1 fputc 1.2 fgetc 二、fputs和fgets函数 2.1 fputs函数 2.2 fgets函数 一、fgetc和fputc函数 1.1 fputc 返回类…

结构体类型,结构体变量的创建和初始化 以及结构中存在的内存对齐

一般结构体类型的声明 struct 结构体类型名 { member-list; //成员表列 }variable-list; //变量表列 例如描述⼀个学⽣&#xff1a; struct Stu { char name[20]; //名字 int age; //年龄 char sex[5]; //性别 }&#xff1b; //结构体变量的初始化 int main() { S…

鸿蒙OS开发实例:【工具类封装-页面路由】

import common from ohos.app.ability.common; import router from ohos.router 封装app内的页面之间跳转、app与app之间的跳转工具类 【使用要求】 DevEco Studio 3.1.1 Release api 9 【使用示例】 import MyRouterUtil from ../common/utils/MyRouterUtil MyRouterUtil…

基于重写ribbon负载实现灰度发布

项目结构如下 代码如下&#xff1a; pom&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocat…

使用第三方远程连接工具ssh连接vagrant创建的虚拟机

vagrant默认密码都是vagrant 密码认证默认是关闭的&#xff0c;进入虚拟机&#xff0c;打开密码认证 1、使用命令vi /etc/ssh/sshd_config进入配置&#xff0c;注意要切换到root用户&#xff0c;这个配置root有权限 2、找到PasswordAuthentication默认为no,改为yes 3、重启虚…

ETL中RESTful API 组件的用法

一、ETL是什么 ETL&#xff0c;全称为Extract-Transform-Load&#xff0c;即数据提取&#xff08;Extract&#xff09;、数据转换&#xff08;Transform&#xff09;和数据加载&#xff08;Load&#xff09;。这是数据仓库中数据处理的重要过程。ETL过程中&#xff0c;数据从源…

小小狠招:巧妙使用HANA数据库的jdbc driver

SAP旗下的HANA数据库&#xff0c;实际上是分为两个系列进行发布&#xff0c;一种是基于本地部署的称之为HANA Platform。另一种是面向Cloud平台的&#xff0c;称之为HANA Cloud。 在实际使用当用&#xff0c;因为两者基本上共用同一代码库&#xff0c;除个别地方略有差异以外&…

【更清晰】照片分享,欢迎家庭新成员HPE ProLiant DL580 Gen9

正文共&#xff1a;1234 字 29 图&#xff0c;预估阅读时间&#xff1a;1 分钟 距离上一台服务器HPE ProLiant DL360 Gen9开箱已经过去4年了&#xff0c;回忆满满&#xff08;风雨同舟&#xff0c;感谢HP Proliant DL360 Gen9陪我走过的四年&#xff09;&#xff1b;就在上周&a…

相册清理大师-手机重复照片整理、垃圾清理软件

相册清理大师是一款超级简单实用的照片视频整理工具。通过便捷的操作手势&#xff0c;帮助你极速整理相册中的照片和视频、释放手机存储空间。 【功能简介】 向上滑动&#xff1a;删除不要的照片 向左滑动&#xff1a;切换下一张照片 向右滑动&#xff1a;返回上一张照片 整理分…

拌合楼管理软件开发(十三) 对接耀华XK3190-A9地磅(实战篇)

前言: 实战开整 目前而言对于整个拌合楼管理软件开发,因为公司对这个项目还处于讨论中,包括个人对其中的商业逻辑也存在一些质疑,都是在做一些技术上的储备.很早就写好了串口与地磅对接获取代码,也大概知道真个逻辑,这次刚好跟库区沟通,远程连接到磅房电脑,开始实操一下. 一、地…

Sql注入---基础

文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 一.Sql注入概述 攻击者通过构造恶意的SQL查询语句&#xff0c;将其注入到应用程序的数据库查询中&#xff0c;以执行未经授权的操作或者获取敏感信息。 假设如下场景&#xff0c;当你想要知道对…

双端队列的插入与删除操作的实现及其时间复杂度分析

双端队列(deque,全称为double-ended queue)是一种支持在两端插入和删除元素的数据结构。与栈和队列不同,双端队列提供了更加灵活的操作方式。在实现双端队列时,我们可以采用数组作为底层数据结构,以保证插入和删除操作的时间复杂度为O(1)。 一、双端队列的基本概念 双…

《QT实用小工具·四》屏幕拾色器

1、概述 源码放在文章末尾 该项目实现了屏幕拾色器的功能&#xff0c;可以根据鼠标指定的位置识别当前位置的颜色 项目功能包含&#xff1a; 鼠标按下实时采集鼠标处的颜色。 实时显示颜色值。 支持16进制格式和rgb格式。 实时显示预览颜色。 根据背景色自动计算合适的前景色…