百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

使用栈实现中缀表达式计算器(用栈实现中缀表达式到后缀表达式的转换)

off999 2024-10-02 18:36 14 浏览 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);
	}

}


相关推荐

Python 数据分析——利用Pandas进行分组统计

话说天下大势,分久必合,合久必分。数据分析也是如此,我们经常要对数据进行分组与聚合,以对不同组的数据进行深入解读。本章将介绍如何利用Pandas中的GroupBy操作函数来完成数据的分组、聚合以及统计...

python数据分析:介绍pandas库的数据类型Series和DataFrame

安装pandaspipinstallpandas-ihttps://mirrors.aliyun.com/pypi/simple/使用pandas直接导入即可importpandasas...

使用DataFrame计算两列的总和和最大值_[python]

【如果对您有用,请关注并转发,谢谢~~】最近在处理气象类相关数据的空间计算,在做综合性计算的时候,DataFrame针对每列的统计求和、最大值等较为方便,对某行的两列或多列数据进行求和与最大值等的简便...

8-Python内置函数

Python提供了丰富的内置函数,这些函数可以直接使用而无需导入任何模块。以下是一些常用的内置函数及其示例:1-print()1-1-说明输出指定的信息到控制台。1-2-例子2-len()2-1-说...

Python中函数式编程函数: reduce()函数

Python中的reduce()函数是一个强大的工具,它通过连续地将指定的函数应用于序列(如列表)来对序列(如列表)执行累积操作。它是functools模块的一部分,这意味着您需要在使用它之...

万万没想到,除了香农计划,Python3.11竟还有这么多性能提升

众所周知,Python3.11版本带来了较大的性能提升,但是,它具体在哪些方面上得到了优化呢?除了著名的“香农计划”外,它还包含哪些与性能相关的优化呢?本文将带你一探究竟!作者:BeshrKay...

最全python3.11版12类75个内置函数大全

获取全部内置函数:importbuiltins#导入模块yc=[]#异常属性nc=[]#不可调用fn=[]#内置函数defll(ty=builtins):...

软件测试笔试题

测试工程师岗位,3-5年,10-14k1.我司有一款产品,类似TeamViewer,向日葵,mstsc,QQ远程控制产品,一个PC客户端产品,请设想一下测试要点。并写出2.写出常用的SQL语句8条,l...

备战各大互联网巨头公司招聘会,最全Python面试大全,共300题

前言众所周知,越是顶尖的互联网公司在面试这一part的要求就越高,需要你有很好的技术功底、项目经验、一份漂亮的简历,当然还有避免不了的笔试过关。对于Python的工程师来说,全面掌握好有关Python...

经典 SQL 数据库笔试题及答案整理

马上又是金三银四啦,有蛮多小伙伴在跳槽找工作,但对于年限稍短的软件测试工程师,难免会需要进行笔试,而在笔试中,基本都会碰到一道关于数据库的大题,今天这篇文章呢,就收录了下最近学员反馈上来的一些数据库笔...

用Python开发日常小软件,让生活与工作更高效!附实例代码

引言:Python如何让生活更轻松?在数字化时代,编程早已不是程序员的专属技能。Python凭借其简洁易学的特点,成为普通人提升效率、解决日常问题的得力工具。无论是自动化重复任务、处理数据,还是开发个...

太牛了!102个Python实战项目被我扒到了!建议收藏!

挖到宝了!整整102个Python实战项目合集,从基础语法到高阶应用全覆盖,附完整源码+数据集,手把手带你从代码小白变身实战大神!这波羊毛不薅真的亏到哭!超全项目库,学练一站式搞定这份资...

Python中的并发编程

1.Python对并发编程的支持多线程:threading,利用CPU和IO可以同时执行的原理,让CPU不会干巴巴等待IO完成。多进程:multiprocessing,利用多核CPU...

Python 也有内存泄漏?

1.背景前段时间接手了一个边缘视觉识别的项目,大功能已经开发的差不多了,主要是需要是优化一些性能问题。其中比较突出的内存泄漏的问题,而且不止一处,有些比较有代表性,可以总结一下。为了更好地可视化内存...

python爬虫之多线程threading、多进程、协程aiohttp批量下载图片

一、单线程常规下载常规单线程执行脚本爬取壁纸图片,只爬取一页的图片。importdatetimeimportreimportrequestsfrombs4importBeautifu...

取消回复欢迎 发表评论: