将中缀表达式转前缀或后缀表达式并计算

参考了http://blog.csdn.net/zhangkunrun/article/details/41699173的解释,代码部分自己完成了,做个小笔记.

举例:
(3 + 4) * 5 - 6 就是中缀表达式
-*+ 3 4 5 6 前缀表达式
3 4 + 5 * 6 - 后缀表达式

这里说点细节:
1.前缀表达式是从右往左扫描的,后缀表达式是从左往右扫描的.
2.前缀表达式遇到右括号直接压入运算符栈的,遇到左括号就会把运算符出栈然后压入中间结果栈,直到遇到左括号,然后把两个括号都丢弃.后缀表达式反之.
3.在遇到运算符的时候,会需要判断本运算符和运算符栈栈顶运算符的优先级,在前缀表达式中是本运算符高或者等于栈顶运算符时将本运算符压入运算符栈;而后缀表达式是本运算符高于栈顶运算符时才将本运算符压入运算符栈.
4.在中缀表达式转前缀或后缀表达式的最后,前缀表达式直接把中间结果栈的元素依次输出就是前缀表达式了,而后缀表达式需要依次输出中间结果栈的元素之后再逆序.
5.在用前缀表达式进行计算的时候,遇到运算符时,弹出栈顶的两个数,用运算符对它们做相应的计算的时候,用栈顶元素 op 次顶元素;;相反后缀表达式的话是用次顶元素 op 栈顶元素

中缀表达式转前缀表达式并计算的代码:

/** 把中缀表达式转前缀表达式后再计算
     * 从右至左扫描表达式,遇到数字时,将数字压入堆栈,遇到运算符时,弹出栈顶的两个数,
     * 用运算符对它们做相应的计算(栈顶元素 op 次顶元素),并将结果入栈;
     * 重复上述过程直到表达式最左端,最后运算得出的值即为表达式的结果。
     * @param zhongExpression
     * @return
     */
    public static int calculateQianZhui(String zhongExpression) {

        //先把中缀表达式转前缀表达式
        String qianExpression = zhong2Qian(zhongExpression);

        //从右至左扫描表达式的话要先把字符串逆序
        qianExpression = reverseString(qianExpression);

        char[] chars = qianExpression.toCharArray();

        Stack<Integer> resultValues = new Stack<>();

        for (char theChar : chars){
            //遇到数字时,将数字压入堆栈,这里要注意字符数字转int,不能直接用integer.valueOf,这样会转成字符的ascii码
            if (Character.isDigit(theChar))
                resultValues.push(Integer.valueOf(String.valueOf(theChar)));
            //遇到运算符时,弹出栈顶的两个数,用运算符对它们做相应的计算(栈顶元素 op 次顶元素),并将结果入栈;
            else {
                int a = resultValues.pop();
                int b = resultValues.pop();
                resultValues.push(calculate(a,b,theChar));
            }
        }
        return resultValues.pop();
    }

    /**
     * 中缀表达式转前缀表达式.
     * (1) 初始化两个栈:运算符栈S1和储存中间结果的栈S2;
     * (2) 从右至左扫描中缀表达式;
     * (3) 遇到操作数时,将其压入S2;
     * (4) 遇到运算符时,比较其与S1栈顶运算符的优先级:
     * (4-1) 如果S1为空,则直接将此运算符入栈;
     * (4-2) 否则,若优先级比栈顶运算符的较高或相等(后缀表达式中是较高,没有相等)或栈顶运算符为右括号“)”,也将运算符压入S1;
     * (4-3) 否则,将S1栈顶的运算符弹出并压入到S2中,再次转到(4-1)与S1中新的栈顶运算符相比较;
     * (5) 遇到括号时:
     * (5-1) 如果是右括号“)”,则直接压入S1;
     * (5-2) 如果是左括号“(”,则依次弹出S1栈顶的运算符,并压入S2,直到遇到右括号为止,此时将这一对括号丢弃;
     * (6) 重复步骤(2)至(5),直到表达式的最左边;
     * (7) 将S1中剩余的运算符依次弹出并压入S2;
     * (8) 依次弹出S2中的元素并输出,结果即为中缀表达式对应的前缀表达式。
     * (后缀表达式这里要将字符串反转输出,这里直接输出即可)
     *
     * @param expression
     * @return
     */
    private static String zhong2Qian(String expression) {

        //(1) 初始化两个栈:运算符栈S1和储存中间结果的栈S2;
        Stack<Character> ops = new Stack<>();
        Stack<Character> resultValues = new Stack<>();

        //(2) 从右至左扫描中缀表达式;这里是先反转字符串再遍历其字符数组
        expression = reverseString(expression);
        char[] chars = expression.toCharArray();
        for (char theChar : chars) {
            //(3) 遇到操作数时,将其压入S2;
            if (Character.isDigit(theChar))
                resultValues.push(theChar);
            //(4) 遇到运算符时,比较其与S1栈顶运算符的优先级:
            else if (theChar == '+' || theChar == '-' || theChar == '*' || theChar == '/') {
                    dealTheChar(ops, resultValues, theChar);

            }
            //(5-1) 如果是右括号“)”,则直接压入S1;
            else if (theChar == ')')
                ops.push(theChar);
            //(5-2) 如果是左括号“(”,则依次弹出S1栈顶的运算符,并压入S2,直到遇到右括号为止,此时将这一对括号丢弃;
            else if (theChar == '(') {
                char opsChar = ops.pop();
                while (opsChar != (')')) {
                    resultValues.push(opsChar);
                    opsChar = ops.pop();
                }
            }
        }

        //(7)将S1中剩余的运算符依次弹出并压入S2;
        while (!ops.empty()) {
            resultValues.push(ops.pop());
        }

        //(8) 依次弹出S2中的元素并输出,结果即为中缀表达式对应的前缀表达式。
        // (后缀表达式这里要将字符串反转输出,这里直接输出即可)
        StringBuilder exp = new StringBuilder();
        while (!resultValues.empty())
            exp.append(resultValues.pop());

        return exp.toString();
    }

    private static String reverseString(String expression) {
        int n = expression.length();
        char[] chars = expression.toCharArray();
        for (int i = 0 ; i < n/2 ; i++) {
            char temp = chars[i];
            chars[i] = chars[n-1-i];
            chars[n-1-i] = temp;
        }
        return String.valueOf(chars);
    }

    private static void dealTheChar(Stack<Character> ops, Stack<Character> resultValues, char theChar) {
        //(4-1) 如果S1为空,则直接将此运算符入栈;
        if (ops.empty()) {
            ops.push(theChar);
            return;
        }
        char popTheChar = ops.pop().charValue();
        //(4-1) 如果S1为空,或栈顶运算符为右括号“)”,则直接将此运算符入栈;
        //(4-2) 否则,若优先级比栈顶运算符的较高或相等(后缀表达式中是较高,没有相等)或栈顶运算符为右括号“)”,,也将运算符压入S1;
        if (popTheChar == ')' || getPriorityValue(theChar) >= getPriorityValue(popTheChar)) {
            ops.push(popTheChar);
            ops.push(theChar);
        }
        //(4-3) 否则,将S1栈顶的运算符弹出并压入到S2中,再次转到(4-1)与S1中新的栈顶运算符相比较;
        else {
            resultValues.push(popTheChar);
            dealTheChar(ops, resultValues, theChar);
        }

    }

    private static int getPriorityValue(char c) {
        if (c == '+' || c == '-')
            return 0;
        if (c == '*' || c == '/')
            return 1;
        throw new RuntimeException("非法操作符");
    }

    private static int calculate(int a,int b,char c) {
        switch (c){
            case '+':
                return a+b;
            case '-':
                return a-b;
            case '*':
                return a*b;
            case '/':
                return a/b;
            default:
                throw new RuntimeException("非法操作符");
        }
    }

中缀表达式转后缀表达式并计算的代码:

/**
     * 后缀表达式的计算机求值:
     * 与前缀表达式类似,只是顺序是从左至右:
     * 从左至右扫描表达式,遇到数字时,将数字压入堆栈,遇到运算符时,
     * 弹出栈顶的两个数,用运算符对它们做相应的计算(次顶元素 op 栈顶元素),
     * 并将结果入栈;重复上述过程直到表达式最右端,最后运算得出的值即为表达式的结果。
     * @param zhongExpression
     * @return
     */
    public static int calculateQianHou(String zhongExpression) {

        //先把中缀表达式转后缀表达式
        String HouExpression = zhong2hou(zhongExpression);

        char[] chars = HouExpression.toCharArray();

        Stack<Integer> resultValues  = new Stack<>();

        for (char theChar : chars){
            //从左至右扫描表达式,遇到数字时,将数字压入堆栈
            if (Character.isDigit(theChar))
                resultValues.push(Integer.valueOf(String.valueOf(theChar)));
            //遇到运算符时,弹出栈顶的两个数,用运算符对它们做相应的计算(次顶元素 op 栈顶元素)
                // (注意前缀表达式计算是栈顶元素 op 次顶元素)
            else {
                int a = resultValues.pop();
                int b = resultValues.pop();
                resultValues.push(calculate(b,a,theChar));
            }
        }
        return resultValues.pop();
    }

    /**
     * 将中缀表达式转换为后缀表达式:
     与转换为前缀表达式相似,遵循以下步骤:
     * (1) 初始化两个栈:运算符栈S1和储存中间结果的栈S2;
     * (2) 从左至右扫描中缀表达式;
     * (3) 遇到操作数时,将其压入S2;
     * (4) 遇到运算符时,比较其与S1栈顶运算符的优先级:
     * (4-1) 如果S1为空,或栈顶运算符为左括号“(”,则直接将此运算符入栈;
     * (4-2) 否则,若优先级比栈顶运算符的高,也将运算符压入S1(注意转换为前缀表达式时是优先级较高或相同,而这里则不包括相同的情况);
     * (4-3) 否则,将S1栈顶的运算符弹出并压入到S2中,再次转到(4-1)与S1中新的栈顶运算符相比较;
     * (5) 遇到括号时:
     * (5-1) 如果是左括号“(”,则直接压入S1;
     * (5-2) 如果是右括号“)”,则依次弹出S1栈顶的运算符,并压入S2,直到遇到左括号为止,此时将这一对括号丢弃;
     * (6) 重复步骤(2)至(5),直到表达式的最右边;
     * (7) 将S1中剩余的运算符依次弹出并压入S2;
     * (8) 依次弹出S2中的元素并输出,结果的逆序即为中缀表达式对应的后缀表达式(转换为前缀表达式时不用逆序)。
     * @return
     */
    private static String zhong2hou(String expression) {

        //(1) 初始化两个栈:运算符栈S1和储存中间结果的栈S2;
        Stack<Character> ops = new Stack<>();
        Stack<Character> resultValues = new Stack<>();

        char[] chars = expression.toCharArray();

        //(2) 从左至右扫描中缀表达式;
        for (char theChar : chars){
            //(3) 遇到操作数时,将其压入S2;
            if (Character.isDigit(theChar))
                resultValues.push(theChar);
            //(4) 遇到运算符时,比较其与S1栈顶运算符的优先级:
            else if (theChar == '+' || theChar == '-' || theChar == '*' || theChar == '/')
                dealTheChar(ops,resultValues,theChar);
            //(5) 遇到括号时:
            //(5-1) 如果是左括号“(”,则直接压入S1;
            else if (theChar == '(')
                ops.push(theChar);
            //(5-2) 如果是右括号“)”,则依次弹出S1栈顶的运算符,并压入S2,直到遇到左括号为止,此时将这一对括号丢弃;
            else if (theChar == ')'){
                char opsChar  = ops.pop();
                while (opsChar != '(') {
                    resultValues.push(opsChar);
                    opsChar  = ops.pop();
                }
            }
        }

        //(7) 将S1中剩余的运算符依次弹出并压入S2;
        while (!ops.isEmpty())
            resultValues.push(ops.pop());
        
        //(8) 依次弹出S2中的元素并输出,结果的逆序即为中缀表达式对应的后缀表达式(转换为前缀表达式时不用逆序)。
        StringBuilder exp = new StringBuilder();
        while (!resultValues.empty())
            exp.append(resultValues.pop());

        return reverseString(exp.toString());
    }

    private static void dealTheChar(Stack<Character> ops, Stack<Character> resultValues, char theChar) {
        //(4-1) 如果S1为空,则直接将此运算符入栈;
        if (ops.empty()) {
            ops.push(theChar);
            return;
        }
        char popTheChar = ops.pop().charValue();
        //(4-1) 如果S1为空,或栈顶运算符为右括号“(”,则直接将此运算符入栈;
        //(4-2) 否则,若优先级比栈顶运算符的高,也将运算符压入S1(注意转换为前缀表达式时是优先级较高或相同,而这里则不包括相同的情况)
        if (popTheChar == '(' || getPriorityValue(theChar) > getPriorityValue(popTheChar)) {
            ops.push(popTheChar);
            ops.push(theChar);
        }
        //(4-3) 否则,将S1栈顶的运算符弹出并压入到S2中
        else {
            resultValues.push(popTheChar);
            //,再次转到(4-1)与S1中新的栈顶运算符相比较;
            dealTheChar(ops, resultValues, theChar);
        }

    }

    private static int getPriorityValue(char c) {
        if (c == '+' || c == '-')
            return 0;
        if (c == '*' || c == '/')
            return 1;
        throw new RuntimeException("非法操作符");
    }

    private static int calculate(int a,int b,char c) {
        switch (c){
            case '+':
                return a+b;
            case '-':
                return a-b;
            case '*':
                return a*b;
            case '/':
                return a/b;
            default:
                throw new RuntimeException("非法操作符");
        }
    }

    private static String reverseString(String string) {

        int n = string.length();
        char[] chars = string.toCharArray();
        for (int i = 0; i < n/2; i++) {
            char temp = chars[i];
            chars[i] = chars[n-1-i];
            chars[n-1-i] = temp;
        }
        return String.valueOf(chars);
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,064评论 5 466
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,606评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,011评论 0 328
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,550评论 1 269
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,465评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 47,919评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,428评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,075评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,208评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,185评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,191评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,914评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,482评论 3 302
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,585评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,825评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,194评论 2 344
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,703评论 2 339

推荐阅读更多精彩内容