使用栈实现中缀表达式计算器(用栈实现中缀表达式到后缀表达式的转换)
off999 2024-10-02 18:36 33 浏览 0 评论
计算器实现原理分析
0. 准备工作:
(1) 一个用来存储数值的数值栈:numStack
(2) 一个用来存储运算符的运算符栈:operStack
1. 将字符串表达式拆分,然后逐个扫描。
(1) 如果是数字,直接压入数值栈
(2) 如果是运算符
1 如果当前运算符栈为空,直接将运算符压入运算符栈
2 如果运算符不为空
1) 如果当前运算符的优先级高于运算符栈栈顶的运算符,则直接压入运算符栈
2) 如果当前运算符的优先级小于或等于运算符栈栈顶的运算符,则需要将栈顶的运算符弹出,同时需要从数值栈中弹出两个数,与栈顶运算与进行运算,然后将运算结果重新压入数值栈。最后继续判断当前运算符与栈顶运算符的优先级,重复该步骤。
2. 表达式扫描完成后,如果运算符栈不为空,则弹出栈顶运算符,同时弹出数值栈中的两个数,进行运算,将结果重新压入数值栈,直到运算符栈为空,此时数值栈中的最后一个数值即为表达式的运算结果。
表达式入栈原理图解
表达式为:2*3*2+4-5-6*2,首先将表达式拆分为
"2","*","3","*","2","+","4","-","5","-","6","*","2"
1. 扫描第一个字符串,判断为数字 "2",直接压入数值栈
2. 扫描第二个字符串,判断为运算符"*",运算符栈为空,直接压入运算符栈
3. 扫描第三个字符串,判断为数字"3",直接压入数值栈
4. 扫描第四个字符串,判断为运算符"*",运算符栈不为空,与栈顶的运算符 * 优先级相等,需要进行运算
1) 首先弹出运算符栈栈顶的 *
2) 弹出数值栈中的 3
3) 弹出数值栈中的 2
4) 将 3 * 2 的结果重新压入数值栈
5) 将第四个字符串的*压入运算符栈
5. 扫描第五个字符串,判断为数值"2",直接压入数值栈
6. 扫描第六个字符串,判断为运算符 "+",继续与运算符栈顶进行优先级比较。结果为小于
1) 首先弹出运算符栈栈顶的 *
2) 弹出数值栈中的 2
3) 弹出数值栈中的 6
4) 将 2 * 6 的结果重新压入数值栈
5) 将第六个字符串的+压入运算符栈
7. 扫描第七个字符串,判断为数值"4",直接入数值栈
8. 扫描第八个字符串,判断为运算符 "-",继续与运算符栈顶进行优先级比较。结果为等于
9. 扫描第九个字符串,判断为数值"5",直接入数值栈
10. 扫描第十个字符串,判断为运算符 "-",继续与运算符栈顶进行优先级比较。结果为等于
1) 首先弹出运算符栈栈顶的 -
2) 弹出数值栈中的 5
3) 弹出数值栈中的 16
4) 将 16 - 5 的结果重新压入数值栈
5) 将第十个字符串的-压入运算符栈
11. 扫描第十一个字符串,判断为数值"6",直接入数值栈
12. 扫描第十二个字符串,判断为运算符 "*",继续与运算符栈顶进行优先级比较。结果为大于
13. 扫描第十三个字符串,判断为数值"2",直接入数值栈
表达式解析入栈过程
计算过程图解
表达式为:2*3*2+4-5-6*2,解析完成后,栈内情况如下图所示
重复如下步骤,直到运算符栈为空:
1) 从 operStack 栈弹出一个运算符
2) 从 numStack 栈弹出两个数据
3) 计算结果后将结果重新压入数值栈
中缀表达式计算器代码实现
/**
* All rights Reserved, Designed By https://www.tulingxueyuan.com/
* @Title: MyCalculator.java
* @Package com.tuling.calculator
* @Description:
* @author: 小白
* @Date 2019年12月3日 下午3:28:03
* @version V1.0
* @Copyright:
*/
package com.tuling.calculator;
import java.util.ArrayList;
import java.util.List;
import java.util.StringTokenizer;
/**
* @ClassName MyCalculator
* @Description 中缀表达式实现计算器
* @Author 北京图灵学院
* @Date 2019年12月3日 下午3:28:03
*/
public class MyCalculator {
//数值栈
private MyStack numStack;
//运算符栈
private MyStack operStack;
//ArrayList 用来存放表达式拆分后的字符串
private List<String> sourceExp;
/**
* @Title: MyCalculator
* @Description:
* @param:
* @throws
*/
public MyCalculator() {
numStack = new MyStack(10);
operStack = new MyStack(10);
sourceExp = new ArrayList<String>();
}
/**
*
* @Title: calculate
* @Description:计算
* @param: @param expression
* @param: @return
* @return: int
* @throws
*/
public int calculate(String expression) {
//先将原始表达式拆分
getExp(expression);
//将原始表达式逐个解析,入栈
inStack(sourceExp);
//入栈完成,依次从运算符栈弹出一个运算符,从数值栈弹出两个数,进行运算。
while(!operStack.isEmpty()) {
//弹出当前运算符栈顶的运算符
String oper = operStack.pop();
String num1 = numStack.pop();
String num2 = numStack.pop();
//计算结果
int result = cal(Integer.parseInt(num1), Integer.parseInt(num2), oper);
//将结果压入数值栈
numStack.push(result+"");
}
//将数值栈的数返回。
return Integer.parseInt(numStack.pop());
}
/**
* @param list
* @Title: inStack
* @Description:将原始表达式逐个解析,进行入栈操作
* 遍历集合,拿到每一个元素并且判断
* 1.如果是运算符
* 1.1> 如果当前运算符栈不为空:
* 1.1.1> 如果当前运算符的优先级小于或者等于栈顶的运算符优先级
* 从数值栈中弹出两个数,并且将运算的结果压入数值栈
* 将运算符压入运算符栈
* 1.1.2> 否则:直接压入运算符栈
* 1.2> 如果当前运算符栈为空:直接压入运算符栈
* 2.如果是数值:
* 直接压入数值栈
* @param:
* @return: void
* @throws
*/
private void inStack(List<String> list) {
for(String str:list) {
if(isOper(str)) {
//判断
if(operStack.isEmpty()) {
//运算符栈为空,直接入栈
operStack.push(str);
}else {
//不为空,需要与栈顶运算符比较优先级高低
//如果当前运算符的优先级小于或等于运算符栈顶的优先级,需要运算
if(priority(str) <= priority(operStack.peek())) {
//需要运算,
//弹出当前运算符栈顶的运算符
String oper = operStack.pop();
String num1 = numStack.pop();
String num2 = numStack.pop();
//计算结果
int result = cal(Integer.parseInt(num1), Integer.parseInt(num2), oper);
//将结果压入数值栈
numStack.push(result+"");
//将运算符压入运算符栈
operStack.push(str);
}else if(priority(str) > priority(operStack.peek())) {
//直接入运算符栈
operStack.push(str);
}
}
}else if(isNum(str)) {
//数字,直接入数值栈
numStack.push(str);
}
}
}
public int cal(int num1,int num2,String oper) {
int result = 0;
switch (oper) {
case "+":
result = num1 + num2;
break;
case "-":
result = num2 - num1;
break;
case "*":
result = num1 * num2;
break;
case "/":
result = num2 / num1;
break;
default:
break;
}
return result;
}
/**
*
* @Title: priority
* @Description:返回运算符的优先级
* @param: @param oper
* @param: @return
* @return: int
* @throws
*/
public int priority(String oper) {
switch (oper) {
case "*":
case "/":
return 2;
case "+":
case "-":
return 1;
default:
return 0;
}
}
/**
* @Title: isNum
* @Description:判断是否为数字
* @param: @param str
* @param: @return
* @return: boolean
* @throws
*/
private boolean isNum(String str) {
return str.matches("[0-9]+");
}
/**
* @Title: isOper
* @Description:判断是否为运算符
* @param: @param str
* @param: @return
* @return: boolean
* @throws
*/
private boolean isOper(String str) {
return str.matches("[\\+\\-\\*\\/]");
}
/**
* @Title: getExp
* @Description:将原始表达式拆分为字符串
* @param: @param expression
* @return: void
* @throws
*/
private void getExp(String expression) {
StringTokenizer stringTokenizer = new StringTokenizer(expression, "+-*/", true);
while(stringTokenizer.hasMoreTokens()) {
sourceExp.add(stringTokenizer.nextToken());
}
}
}
测试类代码如下:
/**
* All rights Reserved, Designed By https://www.tulingxueyuan.com/
* @Title: Test.java
* @Package com.tuling.calculator
* @Description:
* @author: 小白
* @Date 2019年12月3日 下午7:28:33
* @version V1.0
* @Copyright:
*/
package com.tuling.calculator;
/**
* @ClassName Test
* @Description
* @Author 北京图灵学院
* @Date 2019年12月3日 下午7:28:33
*/
public class Test {
/**
* @Title: main
* @Description:
* @param: @param args
* @return: void
* @throws
*/
public static void main(String[] args) {
// String expression = "2+3*4";
String expression = "2*3*4+5-6-4*2";
MyCalculator mc = new MyCalculator();
int result = mc.calculate(expression);
System.out.println("表达式:" + expression + " = " + result);
}
}
相关推荐
- 阿里云国际站ECS:阿里云ECS如何提高网站的访问速度?
-
TG:@yunlaoda360引言:速度即体验,速度即业务在当今数字化的世界中,网站的访问速度已成为决定用户体验、用户留存乃至业务转化率的关键因素。页面加载每延迟一秒,都可能导致用户流失和收入损失。对...
- 高流量大并发Linux TCP性能调优_linux 高并发网络编程
-
其实主要是手里面的跑openvpn服务器。因为并没有明文禁p2p(哎……想想那么多流量好像不跑点p2p也跑不完),所以造成有的时候如果有比较多人跑BT的话,会造成VPN速度急剧下降。本文所面对的情况为...
- 性能测试100集(12)性能指标资源使用率
-
在性能测试中,资源使用率是评估系统硬件效率的关键指标,主要包括以下四类:#性能测试##性能压测策略##软件测试#1.CPU使用率定义:CPU处理任务的时间占比,计算公式为1-空闲时间/总...
- Linux 服务器常见的性能调优_linux高性能服务端编程
-
一、Linux服务器性能调优第一步——先搞懂“看什么”很多人刚接触Linux性能调优时,总想着直接改配置,其实第一步该是“看清楚问题”。就像医生看病要先听诊,调优前得先知道服务器“哪里...
- Nginx性能优化实战:手把手教你提升10倍性能!
-
关注△mikechen△,十余年BAT架构经验倾囊相授!Nginx是大型架构而核心,下面我重点详解Nginx性能@mikechen文章来源:mikechen.cc1.worker_processe...
- 高并发场景下,Spring Cloud Gateway如何抗住百万QPS?
-
关注△mikechen△,十余年BAT架构经验倾囊相授!大家好,我是mikechen。高并发场景下网关作为流量的入口非常重要,下面我重点详解SpringCloudGateway如何抗住百万性能@m...
- Kubernetes 高并发处理实战(可落地案例 + 源码)
-
目标场景:对外提供HTTPAPI的微服务在短时间内收到大量请求(例如每秒数千至数万RPS),要求系统可弹性扩容、限流降级、缓存减压、稳定运行并能自动恢复。总体思路(多层防护):边缘层:云LB...
- 高并发场景下,Nginx如何扛住千万级请求?
-
Nginx是大型架构的必备中间件,下面我重点详解Nginx如何实现高并发@mikechen文章来源:mikechen.cc事件驱动模型Nginx采用事件驱动模型,这是Nginx高并发性能的基石。传统...
- Spring Boot+Vue全栈开发实战,中文版高清PDF资源
-
SpringBoot+Vue全栈开发实战,中文高清PDF资源,需要的可以私我:)SpringBoot致力于简化开发配置并为企业级开发提供一系列非业务性功能,而Vue则采用数据驱动视图的方式将程序...
- Docker-基础操作_docker基础实战教程二
-
一、镜像1、从仓库获取镜像搜索镜像:dockersearchimage_name搜索结果过滤:是否官方:dockersearch--filter="is-offical=true...
- 你有空吗?跟我一起搭个服务器好不好?
-
来人人都是产品经理【起点学院】,BAT实战派产品总监手把手系统带你学产品、学运营。昨天闲的没事的时候,随手翻了翻写过的文章,发现一个很严重的问题。就是大多数时间我都在滔滔不绝的讲理论,却很少有涉及动手...
- 部署你自己的 SaaS_saas如何部署
-
部署你自己的VPNOpenVPN——功能齐全的开源VPN解决方案。(DigitalOcean教程)dockovpn.io—无状态OpenVPNdockerized服务器,不需要持久存储。...
- Docker Compose_dockercompose安装
-
DockerCompose概述DockerCompose是一个用来定义和管理多容器应用的工具,通过一个docker-compose.yml文件,用YAML格式描述服务、网络、卷等内容,...
- 京东T7架构师推出的电子版SpringBoot,从构建小系统到架构大系统
-
前言:Java的各种开发框架发展了很多年,影响了一代又一代的程序员,现在无论是程序员,还是架构师,使用这些开发框架都面临着两方面的挑战。一方面是要快速开发出系统,这就要求使用的开发框架尽量简单,无论...
- Kubernetes (k8s) 入门学习指南_k8s kubeproxy
-
Kubernetes(k8s)入门学习指南一、什么是Kubernetes?为什么需要它?Kubernetes(k8s)是一个开源的容器编排系统,用于自动化部署、扩展和管理容器化应用程序。它...
欢迎 你 发表评论:
- 一周热门
-
-
抖音上好看的小姐姐,Python给你都下载了
-
全网最简单易懂!495页Python漫画教程,高清PDF版免费下载
-
Python 3.14 的 UUIDv6/v7/v8 上新,别再用 uuid4 () 啦!
-
python入门到脱坑 输入与输出—str()函数
-
宝塔面板如何添加免费waf防火墙?(宝塔面板开启https)
-
Python三目运算基础与进阶_python三目运算符判断三个变量
-
(新版)Python 分布式爬虫与 JS 逆向进阶实战吾爱分享
-
失业程序员复习python笔记——条件与循环
-
慕ke 前端工程师2024「完整」
-
飞牛NAS部署TVGate Docker项目,实现内网一键转发、代理、jx
-
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)
