使用Java实现“解析并计算基本算术表达式”

代码的实现逻辑很简单,分别将数字和符号放入两个栈,然后再根据条件出栈并计算表达式即可。示例代码如下:

/**
 * 计算并打印简单表达式
 * @param array 表达式列表,比如:7+3*4*5+2+4-3-1
 */
private static void print(List<String> array){
    for(String string : array){
        //数字栈
        Stack<Double> numStack = new Stack<Double>();
        //符号栈
        Stack<Character> opeStack = new Stack<Character>();
        //读取的每个临时数字
        int n = 0;
        //标识当前读取的是否是数字
        boolean flag = false;
        char[] cs = string.toCharArray();
        for(char temp : cs){
            if (Character.isDigit(temp)) {
                //读取的是数字
                n = 10 * n + Integer.parseInt(String.valueOf(temp));
                //表示已经有数字了
                flag = true;
            } else{
                //数字入栈,且重置标识
                if (flag) {
                    numStack.push((double)n);
                    n = 0;
                    flag = false;
                }
 
                //碰到左括号,符号栈入栈
                if(temp == '('){
                    opeStack.push(temp);
                }else if(temp == ')'){
                    //碰到右括号,符号栈不断出栈,并计算括号里面的表达式
                    while (opeStack.peek() != '('){
                        double result = cal(numStack.pop(), numStack.pop(), opeStack.pop());
                        numStack.push(result);
                    }
                    //把左括号出栈了
                    opeStack.pop();
                }else if (isType(temp) > 0) {
                    // 栈为空直接入栈
                    if (opeStack.isEmpty()) {
                        opeStack.push(temp);
                    } else {
                        // 若符号栈顶元素优先级大于或等于要入栈的符号元素,将数字栈顶元素弹出并计算,然后入栈
                        if (isType(opeStack.peek()) >= isType(temp)) {
                            //计算表达式
                            double result = cal(numStack.pop(), numStack.pop(), opeStack.pop());
                            numStack.push(result);
                        }
                        opeStack.push(temp);
                    }
                }
            }
        }
 
        //如果最后是数字
        if (flag) {
            numStack.push((double)n);
        }
        //符号栈不为空,再次计算
        while (!opeStack.isEmpty()) {
            double result = cal(numStack.pop(), numStack.pop(), opeStack.pop());
            numStack.push(result);
        }
        //打印计算结果
        System.out.println(numStack.pop());
    }
}
 
/**
 * 返回运算符的优先级
 */
private static int isType(char c) {
    if (c == '+' || c == '-') {
        return 1;
    } else if (c == '*' || c == '/') {
        return 2;
    } else {
        return 0;
    }
}
 
/**
 * 运算次序是反的,跟入栈出栈次序有关
 */
private static double cal(double rightNum, double leftNum, char c) {
    if (c == '+') {
        return leftNum + rightNum;
    } else if (c == '-') {
        return leftNum - rightNum;
    } else if (c == '*') {
        return leftNum * rightNum;
    } else if (c == '/') {
        if(rightNum == 0){
            throw new RuntimeException("除数不能为0");
        }else{
            return leftNum / rightNum;
        }
    }else{
        throw new RuntimeException("程序不支持输入的符号");
    }
}
 
 
public static void main(String[] args) {
    List<String> array = new ArrayList<>();
    Scanner scanner = new Scanner(System.in);
 
    //捕获输入
    while (true){
        String string = scanner.nextLine();
        //输入'END'时,结束输入
        if(!"END".equals(string)){
            array.add(string);
        }else{
            break;
        }
    }
 
    //计算并打印
    print(array);
}

测试效果如下:
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

就是不掉头发

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值