表达式求值的两种解法-Java版

124 阅读2分钟

题目大意

 求一个非负整数四则混合运算且含嵌套括号表达式的值,即输入四则混合运算表达式,输出其值。如:

数据保证:(1)保证表达式合法(含除数不为0);(2)保证运算数是非负整数。

# 输入:
1+2*(6/2)-4

# 输出:
3.0

双栈结构实现

 符号优先级定义:')+-*/(';维护两个栈: 符号栈、数字栈;遍历输入串过程中计算:

  1. 数字直接入数字栈。
  2. 符号入栈的情况: a. 符号栈为空时 b. 当前符号优先于栈顶符号时 c. 符号栈顶为(
  3. 需要出栈计算的情况: 符号栈顶是非(的更高优先级符号时
class ExpStack {
    private static final String LEVEL_OPTS = ")+-*/(";

    public double solve(String input) {
        Stack<Character> optStack = new Stack<>();
        Stack<Double> numStack = new Stack<>();
        int i = 0;
        while (i < input.length()) {
            // 如果当前是数字时直接入栈
            if (Character.isDigit(input.charAt(i))) {
                int t = 0;
                for (; i < input.length() && Character.isDigit(input.charAt(i)); i++) {
                    t = t * 10 + input.charAt(i) - '0';
                }
                numStack.push((double) t);
            } else {
                // 如果当前是操作符时:
                // 1.符号栈为空时直接,符号入栈
                // 2.当前符号优先于栈顶符合时,符号入栈
                // 3.符号栈顶为优先级最高的'('时,符号入栈
                // 4.符号栈顶是非'('的更高级运算符时,出栈计算表达式
                if (optStack.isEmpty() || compPriority(input.charAt(i), optStack.peek()) > 0) {
                    optStack.push(input.charAt(i++));
                } else if (optStack.peek() == '(') { // 栈顶为左括号 '('
                    if (input.charAt(i) == ')') { // 当前时')'时,括号匹配完成,弹出'('
                        optStack.pop();
                    } else {
                        optStack.push(input.charAt(i));
                    }
                    i++;
                } else {
                    // 栈顶优先级更高且非 '(' : 运算
                    double top = numStack.pop();
                    numStack.push(calc(numStack.pop(), optStack.pop(), top));
                }
            }
        }
        while (!optStack.isEmpty()) {
            double top = numStack.pop();
            numStack.push(calc(numStack.pop(), optStack.pop(), top));
        }
        return numStack.pop();
    }

    private int compPriority(char c1, char c2) {
        return LEVEL_OPTS.indexOf(c1) - LEVEL_OPTS.indexOf(c2);
    }

    private double calc(double x, char o, double y) {
        switch (o) {
            case '+':
                return x + y;
            case '-':
                return x - y;
            case '*':
                return x * y;
            case '/':
                return x / y;
        }
        return 0;
    }
}

二叉树结构实现

 构建二叉树:将符号作为非叶子节点,数字作为叶子结点。最后通过后序变量计算结果。使用二分法建树(根节点是表达式中最后一个计算的运算符,即后缀式中最后一个运算符):

  1. 先排除括号(因为括号要看成一个整体)
  2. 优先取 +-(因为优先级更低,计算时间更晚)
  3. 再考虑 */(没有加减再考虑乘除)
class ExpTree {
    private String mInput;
    private java.util.LinkedList<Node> mTree;

    public double solve(String input) {
        mInput = input;
        mTree = new java.util.LinkedList<>();

        buildTree(0, mInput.length());

        return dfs(mTree.size() - 1);
    }


    private int buildTree(int li, int ri) {

        try { // 先尝试吧表达式解析为叶子节点(纯运算数)
            int n = Integer.parseInt(mInput.substring(li, ri));
            Node node = new Node(n, -1, -1);
            mTree.addLast(node);
            return mTree.size() - 1;
        } catch (Exception ignore) {
        }

        // 找到最外层的运算符(最后一个计算的运算符,优先级最低的符号 opt)
        int opt, as = -1, md = -1, bracket = 0;
        for (int i = li; i < ri; i++) {
            switch (mInput.charAt(i)) {
                case '(':
                    bracket++;
                    break;
                case ')':
                    bracket--;
                    break;
                case '+':
                case '-':
                    if (bracket == 0) {
                        as = i;
                    }
                    break;
                case '*':
                case '/':
                    if (bracket == 0) {
                        md = i;
                    }
                    break;
            }
        }
        opt = as >= 0 ? as : md : as;
        if (opt < 0) { // 发现这是一个被括号包裹的表达式(去掉括号重新构造)
            return buildTree(li + 1, ri - 1);
        }
        // 依次构造左右子树
        Node node = new Node(mInput.charAt(opt), buildTree(li, opt), buildTree(opt + 1, ri));
        mTree.addLast(node);
        return mTree.size() - 1;
    }

    private double dfs(int i) { // 后序遍历求解
        if (mTree.get(i).lch == -1 && mTree.get(i).rch == -1) {
            return mTree.get(i).num;
        }
        switch (mTree.get(i).opt) {
            case '+':
                return dfs(mTree.get(i).lch) + dfs(mTree.get(i).rch);
            case '-':
                return dfs(mTree.get(i).lch) - dfs(mTree.get(i).rch);
            case '*':
                return dfs(mTree.get(i).lch) * dfs(mTree.get(i).rch);
            case '/':
                return dfs(mTree.get(i).lch) / dfs(mTree.get(i).rch);
        }
        return 0;
    }

    private static class Node {
        double num;
        char opt;
        int lch, rch;

        Node(double n, int l, int r) {
            num = n;
            initChild(l, r);
        }

        Node(char o, int l, int r) {
            opt = o;
            initChild(l, r);
        }

        private void initChild(int l, int r) {
            lch = l;
            rch = r;
        }
    }
}

测试驱动

public class Main {
    public static void main(String[] args) {
        Scanner cin = new Scanner(System.in);
        String input = cin.nextLine();

        double t = new ExpTree().solve(input);
        System.out.println(t);

        double s = new ExpStack().solve(input);
        System.out.println(s);
    }
}