数据结构与算法之完整逆波兰计算器

数据结构与算法之完整逆波兰计算器

源码如下:

package com.datastrucate.stack;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import java.util.Stack;

import java.util.regex.Pattern;


/**
 * ClassName:ReversePolishMultiCalc
 * Package:com.datastrucate.stack
 * Description:
 *
 * @Date:2021/5/12 16:46
 * @Author:hm
 */
public class ReversePolishMultiCalc {

    /** * 匹配 + - * / ( ) 运算符 */

    static final String SYMBOL = "\\+|-|\\*|/|\\(|\\)";

    static final String LEFT = "(";
    static final String RIGHT = ")";
    static final String ADD = "+";
    static final String MINUS= "-";
    static final String TIMES = "*";
    static final String DIVISION = "/";


    /** * 加減 + - */

    static final int LEVEL_01 = 1;


    /** * 乘除 * / */

    static final int LEVEL_02 = 2;


    /**

     * 括号

     */

    static final int LEVEL_HIGH = Integer.MAX_VALUE;


    static Stack<String> stack = new Stack<>();

    static List<String> data = Collections.synchronizedList(new ArrayList<String>());



    /**

     *	去除所有空白符

     *	@param s

     *	@return

     */

    public static String replaceAllBlank(String s ){
        // \\s+ 匹配任何空白字符,包括空格、制表符、换页符等等, 等价于[ \f\n\r\t\v]
        return s.replaceAll("\\s+","");

    }


    /**

     * 判断是不是数字 int double long float

     *	@param s

     *	@return

     */

    public static boolean isNumber(String s){

        Pattern pattern = Pattern.compile("^[-\\+]?[.\\d]*$");
        return pattern.matcher(s).matches();

    }



    /**

     *	判断是不是运算符

     *	@param s

     *	@return

     */

    public static boolean isSymbol(String s){

        return s.matches(SYMBOL);

    }



    /**

     *	匹配运算等级

     *	@param s

     *	@return

     */

    public static int calcLevel(String s){
        if("+".equals(s) || "-".equals(s)){
            return LEVEL_01;

        } else if("*".equals(s) || "/".equals(s)){

            return LEVEL_02;

        }

        return LEVEL_HIGH;

    }



    /**

     *	匹配

     *	@param s

     *	@throws Exception */

    public static List<String> doMatch (String s) throws Exception{

        if(s == null || "".equals(s.trim()))
            throw new RuntimeException("data is empty");
        if(!isNumber(s.charAt(0)+""))
            throw new RuntimeException("data illeagle,start not with a number");

        s = replaceAllBlank(s);

        String each; int start = 0;

        for (int i = 0; i < s.length(); i++) {
            if(isSymbol(s.charAt(i)+"")){
                each = s.charAt(i)+"";

                //栈为空,(操作符,或者 操作符优先级大于栈顶优先级 &&  操作符优先级不是( )的优先级 及是 ) 不能直接入栈

                if(stack.isEmpty() || LEFT.equals(each)

                    || ((calcLevel(each) > calcLevel(stack.peek()))
                        && calcLevel(each) < LEVEL_HIGH)){

                    stack.push(each);

                }else if( !stack.isEmpty() && calcLevel(each) <= calcLevel(stack.peek())){ //栈非空,操作符优先级小于等于栈顶优先级时出栈入列,直到栈为空,或者遇到了(,最后操作符入栈

                     while (!stack.isEmpty() && calcLevel(each) <= calcLevel(stack.peek()) ){
                         if(calcLevel(stack.peek()) == LEVEL_HIGH){

                            break;

                         }

                         data.add(stack.pop());

                     }

                     stack.push(each);

                }else if(RIGHT.equals(each)){ // ) 操作符,依次出栈入列直到空栈或者遇到了第一个)操作符,此时)出栈

                    while (!stack.isEmpty() && LEVEL_HIGH >= calcLevel(stack.peek())){
                        if(LEVEL_HIGH == calcLevel(stack.peek())){
                            stack.pop();
                            break;
                        }

                        data.add(stack.pop());

                    }

                }

                start = i ; //前一个运算符的位置
            }else if( i == s.length()-1 || isSymbol(s.charAt(i+1)+"") ){

                each = start == 0 ? s.substring(start,i+1) : s.substring(start+1,i+1);
                if(isNumber(each)) {

                    data.add(each);
                    continue;

                }

                throw new RuntimeException("data not match number");

            }

        }

        //如果栈里还有元素,此时元素需要依次出栈入列,可以想象栈里剩下栈顶为/,栈底为+,应该依次出栈入列,可以直接翻转整个 stack 添加到队列

        Collections.reverse(stack);

        data.addAll(new ArrayList<>(stack));

        System.out.println(data); return data;

    }



    /**

     *	算出结果

     *	@param list

     *	@return

     */

    public static Double doCalc(List<String> list){

        Double d = 0d;

        if(list == null || list.isEmpty()){
            return null;

        }

        if (list.size() == 1){
            System.out.println(list);

            d = Double.valueOf(list.get(0));

            return d;

        }

        ArrayList<String> list1 = new ArrayList<>();
        for (int i = 0; i < list.size(); i++) {

            list1.add(list.get(i));

            if(isSymbol(list.get(i))){

                Double d1 = doTheMath(list.get(i - 2), list.get(i - 1), list.get(i)); list1.remove(i);

                list1.remove(i-1);

                list1.set(i-2,d1+"");

                list1.addAll(list.subList(i+1,list.size()));

                break;

            }

        }

        doCalc(list1);

        return d;

    }



    /**

     *	运算

     *	@param s1

     *	@param s2

     *	@param symbol

     *	@return

     */



    public static Double doTheMath(String s1,String s2,String symbol){

        Double result ;

        switch (symbol){

            case ADD : result = Double.valueOf(s1) + Double.valueOf(s2); break;

            case MINUS : result = Double.valueOf(s1) - Double.valueOf(s2); break;

            case TIMES : result = Double.valueOf(s1) * Double.valueOf(s2); break;

            case DIVISION : result = Double.valueOf(s1) / Double.valueOf(s2); break;

            default : result = null;

        }

        return result;



    }



    public static void main(String[] args) {

        //String math = "9+(3-1)*3+10/2";

        String math = "12.8 + (2 - 3.55)*4+10/5.0";
        try {

            doCalc(doMatch(math));
        } catch (Exception e) {

            e.printStackTrace();

        }

    }


}


  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值