工作之余,突然想用java实现一下四则运算,经过简单的构思,很快就有了思路,就完成了一个简单版本。
经过慎重考虑,觉得使用栈还是要比数组或者List容易对符号和数字进行控制。
我使用两个栈,分别保存数字和符号,然后判断当前符号和前一个符号的运算级别,来决定是否进行一次弹栈运算(就是挨着前一个运算符号的两个数,是不是有必要进行一次运算)。
经过简单的测试,标准的四则运算,在下面代码是没有问题的。
如果再加括号,那就是在这个基础上,添加去括号的流程就可以,等那天有时间了,再完善吧。
所有源码如下:
package com.rq.spring.main;
import java.util.Stack;
/**
* 使用栈完成的简单四则运算,不带有任何括号
* 测试时,请输入标准的四则运算式
* 待优化的地方很多,各位大神,多提意见
* @author 任强
*
*/
public class TestStackSimpleOperation {
public static void main(String[] args) {
//简单语法
// Stack<String> stack = new Stack<>();//创建一个栈,字符串类型
//
// stack.push("a");//压栈
// stack.push("b");
// stack.push("c");
//
// while(!stack.empty()){//判断是否为空
// String s = stack.pop();//弹栈
// System.out.println(s);
// }
//另一种遍历栈的方法
// Iterator<String> it = stack.iterator();
// while(it.hasNext()){
// String s = it.next();
// System.out.println(s);
// }
//测试自己编写的方法
// System.out.println(simpleOperation("1+2+3*5+4*6-7/7+1"));//42
// System.out.println(simpleOperation("1*2*3*4*5*6+1+2*5+9*6+9"));//794
// System.out.println(simpleOperation("5+6/3*8-9+45*67+95/5+8-85"));//2969
System.out.println(simpleOperation("1+2+3-4+5*6-7+8-9*6/2/1+99/33*55-951"));//-780
}
/**
* 简单四则运算
* @param equation 四则运算式
* @return 运算结果
*/
public static int simpleOperation(String equation){
Stack<Integer> nums = new Stack<>();//保存运算数字的栈
Stack<String> symbols = new Stack<>();//保存运算符号的栈
while(equation != null && equation.length() > 0){
int symIndex = getLastIndex(equation);
if(symIndex == -1){//表示已经到了运算式的结尾,需要返回结果了
int num = Integer.valueOf(equation);
int r = getResult(num, symbols.pop(), nums.pop());
while(!symbols.empty()) {
r = getResult(r, symbols.pop(), nums.pop());
}
return r;
}
int num = Integer.valueOf(equation.substring(0, symIndex));
nums.push(num);
String sym = equation.substring(symIndex, symIndex + 1);
//完成计算流程
String lastSym = symbols.empty() ? null : symbols.peek();//获取前一个运算符
//前一个运算符存在,且它的等级是不小于当前运算符,就进行一次运算,
//把数字栈中的后两个元素弹栈操作,使用前一个运算符完成计算
//使用while的原因是可能在符号栈中需要继续进行往前一步的运算
while(lastSym != null && getLevel(lastSym) >= getLevel(sym)){
nums.push(getResult(nums.pop(), symbols.pop(), nums.pop()));
lastSym = symbols.empty() ? null : symbols.peek();
}
symbols.push(sym);
equation = equation.substring(symIndex + 1);
}
return 0;
}
//计算结果,因为是弹栈操作,所有后弹出的数字在运算符前面
private static int getResult(int a, String sym, int b){
switch(sym){
case "+" : return b + a;
case "-" : return b - a;
case "*" : return b * a;
default : return b / a;
}
}
//运算符等级,等级越高需要越先完成计算
private static int getLevel(String sym){
return sym.equals("+") || sym.equals("-") ? 1 : 2 ;
}
//最接近运算式起始的符号位置,用来确定马上要参与运算的数字和该数字之后的运算符
private static int getLastIndex(String equation){
int addIndex = equation.indexOf("+");
int symIndex = -1;
if(addIndex > -1){
symIndex = addIndex;
}
int subIndex = equation.indexOf("-");
if(subIndex > -1 && (symIndex == -1 || symIndex > subIndex)){
symIndex = subIndex;
}
int mulIndex = equation.indexOf("*");
if(mulIndex > -1 && (symIndex == -1 || symIndex > mulIndex)){
symIndex = mulIndex;
}
int divIndex = equation.indexOf("/");
if(divIndex > -1 && (symIndex == -1 || symIndex > divIndex)){
symIndex = divIndex;
}
return symIndex;
}
}