首页 > 其他分享 >计算器思想-中缀表达式转化为后缀表达式

计算器思想-中缀表达式转化为后缀表达式

时间:2023-09-14 23:25:59浏览次数:53  
标签:arr String 后缀 栈顶 list 括号 stack 表达式 中缀

计算机思维和人的思维的不同

对于一个算式3+2*(4-3)/5
人的思维是根据括号和符号优先级,优先计算括号中的数据,在进行乘法和除法,在处理加法运算
但是计算机的思维是线性的,计算机会按照算式的前后顺序,从前往后进行运算,这样会导致运算结果错误

计算机如何套用人的运算思维

想要让计算机具有人的”思维“,就需要使用栈,将数据和运算符号之间的顺序按照计算机可以理解的方式排列
想要改变规则,就需要将人理解的中缀表达式转换为后缀表达式
转化的规则是:

  • 中缀表达式转化成后缀表达式
    1.遇到操作数直接放入到集合中
    2.遇到操作符
    2.1当栈为空,或栈顶元素为(,直接放入到栈中
    2.2当优先级比栈顶元素高时,直接进栈
    2.3当优先级小于等于栈顶元素,将栈顶元素出栈放入到集合中,再和栈顶元素比较
    3.遇到左右括号
    3.1如果为左括号,直接加入栈
    3.2如果为右括号,依次将栈顶元素弹出,直到左括号,并将左括号弹出
    4.最后将栈顶元素依次弹出

中缀表达式转换为后缀表达式的过程

以运算算式 3-(4-3)/5*10
以中缀表达式表示为 3 - 4 - 3 / 5 * 10
后缀表达式表示为 3 4 3 - 5 / 10 * -
其中涉及到了栈的入栈和出栈,

转化过程:
1.先创建一个集合,用于记录后缀表达式,创建一个栈,用于记录运算符
2.3进入集合,-进入栈 集合 3 栈 -
3.左括号进栈 集合3 栈- (
4.4进入集合,-进入栈,与此时的栈顶元素比较,此时栈顶元素是左括号,-直接放入栈中 集合3 4 栈 - ( -
5.3进入集合,-和栈顶元素比较,优先级相等或者小于,将栈顶元素-弹出 集合中为 3 4 3 - 栈-(
6.右括号进栈 将栈顶元素弹出,直到左括号并弹出左括号 集合3 4 3 - 栈-
7./进栈,优先级大于-,直接进栈,5 进入集合 集合3 4 3 - 5 栈- /
8.*进栈,优先等于栈顶元素,弹出栈顶元素,10进入集合 集合 3 4 3 - 5 / 10 栈 - *
9.算式获取完成,将栈中元素全部弹出 集合 3 4 3 - 5 / 10 * -

实现

整数算式运算

package com.prettyspider.calculate;

import java.util.*;


/**
 * @author prettyspider
 * @ClassName CalcInt
 * @description: 传入整数运算字符串,计算其数值
 * @date 2023/9/11 6:21
 * @Version V1.0
 */

public class CalcInt {
    /**
     * 中缀表达式转化成后缀表达式
     * <p>
     * 1.遇到操作数直接放入到集合中
     * 2.遇到操作符
     * 2.1当栈为空,或栈顶元素为(,直接放入到栈中
     * 2.2当优先级比栈顶元素高时,直接进栈
     * 2.3当优先级小于等于栈顶元素,将栈顶元素出栈放入到集合中,再和栈顶元素比较
     * 3.遇到左右括号
     * 3.1如果为左括号,直接加入栈
     * 3.2如果为右括号,依次将栈顶元素弹出,直到左括号,并将左括号弹出
     * 4.最后将栈顶元素依次弹出
     */
    public static void main(String[] args) {
        // 要传入的运算字符串
        String s = "10+2*(3-4)+20*(3*4+3)/3+20";
        // 中缀表达式转化成后缀表达式
        List<String> list = inToPastExpression(s);
        // 计算
        int number = calc(list);
        System.out.println("计算的数值为"+number);
    }


    /**
     * 计算后缀表达式
     * @param list 后缀表达式集合
     * @return 传入的整数运算字符串的计算数值
     */
    private static int calc(List<String> list) {
        // 创建栈,用于记录数据
        Stack<String> stack = new Stack<>();
        for (String s : list) {
            // 1.放入 当是数值时,直接放入栈中
            if (s.matches("\\d+")) {
                stack.push(s);
            } else {
                // 2.去除 当是运算符时,再栈中取出两个数,进行运算,并将运算之后的结果放入到栈中
                int num1, num2;
                switch (s) {
                    case "+":
                        num1 = Integer.parseInt(stack.pop());
                        num2 = Integer.parseInt(stack.pop());
                        stack.push(String.valueOf(num2 + num1));
                        break;
                    case "-":
                        num1 = Integer.parseInt(stack.pop());
                        num2 = Integer.parseInt(stack.pop());
                        stack.push(String.valueOf(num2 - num1));
                        break;
                    case "*":
                        num1 = Integer.parseInt(stack.pop());
                        num2 = Integer.parseInt(stack.pop());
                        stack.push(String.valueOf(num2 * num1));
                        break;
                    case "/":
                        num1 = Integer.parseInt(stack.pop());
                        num2 = Integer.parseInt(stack.pop());
                        stack.push(String.valueOf(num2 / num1));
                        break;
                }
            }
        }
        return Integer.parseInt(stack.pop());
    }


    /**
     * 将传入的字符串进行切分,存放到集合中
     * @param str 传入的算数字符串
     * @return 后缀表达式集合
     */
    private static List<String> inToPastExpression(String str) {
        // 创建集合和栈
        List<String> list = new ArrayList<>();
        Stack<String> stack = new Stack<>();
        // 切分数据
        ArrayList<String> arr = cutString(str);
        for (int i = 0; i < arr.size(); i++) {
            //       * 1.遇到操作数直接放入到集合中
            String value = arr.get(i);
            if (value.matches("\\d+")) {
                list.add(value);
            }
            //      * 3.遇到左右括号
            //      *      3.1如果为左括号,直接加入群
            else if ("(".equals(value)) {
                stack.push(value);
            }
            //      *      3.2如果为右括号,依次将栈顶元素弹出,直到左括号,并将左括号弹出
            else if (")".equals(value)) {
                while (!"(".equals(stack.peek())) {
                    list.add(stack.pop());
                }
                stack.pop();
            }
            //      * 2.遇到操作符
            else {
                //      *      2.1当栈为空,或栈顶元素为(,直接放入到栈中
                //      *      2.3当优先级小于等于栈顶元素,将栈顶元素出栈放入到集合中,再和栈顶元素比较
                while (stack.size() != 0 && !judgeOperator(value, stack.peek())) {
                    list.add(stack.pop());
                }
                //      *      2.2当优先级比栈顶元素高时,直接进栈
                stack.push(value);
            }
        }
        //      * 4.最后将栈顶元素依次弹出
        while (!stack.empty()) {
            list.add(stack.pop());
        }
        System.out.println(list);
        return list;
    }

    /**
     * 将运算字符切分切分成集合
     * @param str 待切分字符串
     * @return ArrayList<String> 被切分之后的字符串集合
     */
    private static ArrayList<String> cutString(String str) {
        String[] arr = new String[str.length()-1];
        // 为字符串数组赋初值
        for (int i = 0; i < arr.length; i++) {
            arr[i] = "";
        }
        int index = 0;
        arr[index] = String.valueOf(str.charAt(0));
        for (int i = 1; i < str.length(); i++) {
            // 1.是数值放入
            if (Character.isDigit(str.charAt(i))) {
                // 1.1并看其起一个数据是不是也数值,是数值,将其前一个数据*10+本身
                if (arr[index].matches("\\d+")) {
                    arr[index] = String.valueOf(Integer.parseInt(arr[index]) * 10 + Integer.parseInt(String.valueOf(str.charAt(i))));
                } else {
                    arr[++index] = String.valueOf(str.charAt(i));
                }
            } else {
                // 2.不是数值,直接加入到集合中
                arr[++index] = String.valueOf(str.charAt(i));
            }
        }
        // 去除null
        ArrayList<String> list = removeNull(arr);
        return list;

    }

    /**
     * 将字符串数组中为空的元素去除
     * @param arr 字符串数组
     * @return 去除控制的字符串集合
     */
    private static ArrayList<String> removeNull(String[] arr) {
        ArrayList<String> list = new ArrayList<>();
        for (String s : arr) {
            if (!"".equals(s)) {
                list.add(s);
            }
        }
        return list;
    }

    /**
     * 比较新旧操作符的权重,判断是存放还是取出
     * @param s1 新的操作符
     * @param s2 旧的操作符
     * @return 新旧操作符的权重比较
     */
    private static boolean judgeOperator(String s1, String s2) {
        int num1 = 0, num2 = 0;
        switch (s1) {
            case "+":
            case "-":
                num1 = 1;
                break;
            case "*":
            case "/":
                num1 = 2;
                break;
        }
        switch (s2) {
            case "+":
            case "-":
                num2 = 1;
                break;
            case "*":
            case "/":
                num2 = 2;
                break;
        }
        // 判断新旧操作符的优先级
        if (num1 > num2) {
            return true;
        }
        return false;
    }
}

小数算式运算

小数运算要考虑小数点,相对于整数要比较的复杂
在获取娴熟时,需要对小数点的位置进行判断,对小数点右边的数据进行处理

package com.prettyspider.calculate;

import java.util.*;


/**
 * @author prettyspider
 * @ClassName CalcInt
 * @description: 传入整数运算字符串,计算其数值
 * @date 2023/9/11 6:21
 * @Version V1.0
 */

public class CalcDouble {
    /**
     * 中缀表达式转化成后缀表达式
     * <p>
     * 1.遇到操作数直接放入到集合中
     * 2.遇到操作符
     * 2.1当栈为空,或栈顶元素为(,直接放入到栈中
     * 2.2当优先级比栈顶元素高时,直接进栈
     * 2.3当优先级小于等于栈顶元素,将栈顶元素出栈放入到集合中,再和栈顶元素比较
     * 3.遇到左右括号
     * 3.1如果为左括号,直接加入栈
     * 3.2如果为右括号,依次将栈顶元素弹出,直到左括号,并将左括号弹出
     * 4.最后将栈顶元素依次弹出
     */
    public static void main(String[] args) {
        // 要传入的运算字符串
        String s = "10.32*3.23+2.234";
        // 中缀表达式转化成后缀表达式
        List<String> list = inToPastExpression(s);
        // 计算
        double number = calc(list);
        System.out.println("计算的数值为"+number);
    }


    /**
     * 计算后缀表达式
     * @param list 后缀表达式集合
     * @return 传入的整数运算字符串的计算数值
     */
    private static double calc(List<String> list) {
        // 创建栈,用于记录数据
        Stack<String> stack = new Stack<>();
        for (String s : list) {
            // 1.放入 当是数值时,直接放入栈中
            if (s.matches("(\\d|\\.)+")) {
                stack.push(s);
            } else {
                // 2.去除 当是运算符时,再栈中取出两个数,进行运算,并将运算之后的结果放入到栈中
                double num1, num2;
                switch (s) {
                    case "+":
                        num1 = Double.valueOf(stack.pop());
                        num2 = Double.valueOf(stack.pop());
                        stack.push(String.valueOf(num2 + num1));
                        break;
                    case "-":
                        num1 = Double.valueOf(stack.pop());
                        num2 = Double.valueOf(stack.pop());
                        stack.push(String.valueOf(num2 - num1));
                        break;
                    case "*":
                        num1 = Double.valueOf(stack.pop());
                        num2 = Double.valueOf(stack.pop());
                        stack.push(String.valueOf(num2 * num1));
                        break;
                    case "/":
                        num1 = Double.valueOf(stack.pop());
                        num2 = Double.valueOf(stack.pop());
                        stack.push(String.valueOf(num2 / num1));
                        break;
                }
            }
        }
        return Double.valueOf(stack.pop());
    }


    /**
     * 将传入的字符串进行切分,存放到集合中
     * @param str 传入的算数字符串
     * @return 后缀表达式集合
     */
    private static List<String> inToPastExpression(String str) {
        // 创建集合和栈
        List<String> list = new ArrayList<>();
        Stack<String> stack = new Stack<>();
        // 切分数据
        ArrayList<String> arr = cutString(str);
        for (int i = 0; i < arr.size(); i++) {
            //       * 1.遇到操作数直接放入到集合中
            String value = arr.get(i);
            if (value.matches("(\\d|\\.)+")) {
                list.add(value);
            }
            //      * 3.遇到左右括号
            //      *      3.1如果为左括号,直接加入群
            else if ("(".equals(value)) {
                stack.push(value);
            }
            //      *      3.2如果为右括号,依次将栈顶元素弹出,直到左括号,并将左括号弹出
            else if (")".equals(value)) {
                while (!"(".equals(stack.peek())) {
                    list.add(stack.pop());
                }
                stack.pop();
            }
            //      * 2.遇到操作符
            else {
                //      *      2.1当栈为空,或栈顶元素为(,直接放入到栈中
                //      *      2.3当优先级小于等于栈顶元素,将栈顶元素出栈放入到集合中,再和栈顶元素比较
                while (stack.size() != 0 && !judgeOperator(value, stack.peek())) {
                    list.add(stack.pop());
                }
                //      *      2.2当优先级比栈顶元素高时,直接进栈
                stack.push(value);
            }
        }
        //      * 4.最后将栈顶元素依次弹出
        while (!stack.empty()) {
            list.add(stack.pop());
        }
        return list;
    }

    /**
     * 将运算字符切分切分成集合
     * @param str 待切分字符串
     * @return ArrayList<String> 被切分之后的字符串集合
     */
    private static ArrayList<String> cutString(String str) {
        String[] arr = new String[str.length()-1];
        // 为字符串数组赋初值
        for (int i = 0; i < arr.length; i++) {
            arr[i] = "";
        }
        int index = 0;
        arr[index] = String.valueOf(str.charAt(0));
        for (int i = 1; i < str.length(); i++) {
            // 1.是数值放入
            if (Character.isDigit(str.charAt(i))|| ".".equals(String.valueOf(str.charAt(i)))) {
                // 1.1并看其起一个数据是不是也数值,是数值,将其前一个数据拼接
                if (arr[index].matches("(\\d|\\.)+")) {
                    arr[index] = arr[index] + str.charAt(i);
                } else {
                    arr[++index] = String.valueOf(str.charAt(i));
                }
            } else {
                // 2.不是数值,直接加入到集合中
                arr[++index] = String.valueOf(str.charAt(i));
            }
        }
        // 去除null
        ArrayList<String> list = removeNull(arr);
        return list;

    }

    /**
     * 将字符串数组中为空的元素去除
     * @param arr 字符串数组
     * @return 去除控制的字符串集合
     */
    private static ArrayList<String> removeNull(String[] arr) {
        ArrayList<String> list = new ArrayList<>();
        for (String s : arr) {
            if (!"".equals(s)) {
                list.add(s);
            }
        }
        return list;
    }

    /**
     * 比较新旧操作符的权重,判断是存放还是取出
     * @param s1 新的操作符
     * @param s2 旧的操作符
     * @return 新旧操作符的权重比较
     */
    private static boolean judgeOperator(String s1, String s2) {
        int num1 = 0, num2 = 0;
        switch (s1) {
            case "+":
            case "-":
                num1 = 1;
                break;
            case "*":
            case "/":
                num1 = 2;
                break;
        }
        switch (s2) {
            case "+":
            case "-":
                num2 = 1;
                break;
            case "*":
            case "/":
                num2 = 2;
                break;
        }
        // 判断新旧操作符的优先级
        if (num1 > num2) {
            return true;
        }
        return false;
    }
}

在获取数据时,巧妙的使用正则获取对应的数据,利于数据的处理

标签:arr,String,后缀,栈顶,list,括号,stack,表达式,中缀
From: https://www.cnblogs.com/prettyspider/p/17703787.html

相关文章

  • 解密MySQL中强大的武器——REGEXP正则表达式
    家人们,今天我来为大家介绍一项在MySQL中非常强大的武器——REGEXP正则表达式。MySQL作为一款广泛使用的关系型数据库管理系统,其内置的REGEXP关键字为我们提供了强大的正则表达式功能,使得我们可以更加灵活和高效地进行数据匹配和处理。以下是一些常见的用法和语法规则来详解REG......
  • 迭代器总结、生成器、生成器表达式、常用的68个内置函数
    迭代器总结(迭代取值和索引取值的对比)#迭代器主要就是一个迭代取值,另外一种取值方式就是索引(下标)取值l=[1,2,3,4]res=l.__iter__()res.__next__()#1res.__next__()#2res1=l.__iter__()res1.__next__()#1res1.__next__()#2迭代取值 1.不依赖......
  • 访问前台页面${pageContext.request.contextPath}/el表达式失效问题解决
    最近在做项目整合这个问题,然后在项目整合的时候,遇到了好多问题,这是其中一个,在此留作记录吧,虽然关键点不是我处理好的。访问前端页面,我先描述一下具体出现的现象:我访问前端jsp页面的时候,jquery文件,js,css样式等都会失效,也就是没有引入到jsp页面当中。查看浏览器console的时候,发现${pa......
  • Lambda表达式
    Lambda表达式要使用lambda表达式就要要使用java8,使用Lambda表达式可以让我们的代码更少,看上去更简洁;它是为了简化了函数式接口匿名内部类的语法。Lambda只能接受函数式接口,所谓的函数式接口指的是只能有一个抽象方法的接口。Lambda表达式语法Lambda表达式通过操作符->分为......
  • 求表达式的值
    利用栈求表达式的值,用Java实现如下所示://栈的定义publicclassArrayStack{//栈的大小privateintSIZE=0;//一维数组int[]arrayStack;//栈顶指针privateinttop=-1;//等于-1时表示栈中没有任何数据//初始化栈的大小publ......
  • Spring小技巧--计算表达式的值
    平时工作中经常要用到表达式值的计算问题,Spring框架中提供了SpringExpressionLanguage(简称SpEL)机制,可以很方便快捷的实现表达式值的计算;SpEL机制需要引入Spring-expression包。下面列举其应用的两个小Demo;1、数值计算:StringexpressionStr="19+26";ExpressionParse......
  • Lambda表达式中的in操作
    //首先,在程序中接受一个数组例如:int[]s=[1,2,3];//在Lamda表达式中使用如下:db.userinfo.where(u=>s.Contains(u.id));//等同于sql语句:select*fromuserinfowhereidin(1,2,3)原文链接:https://blog.csdn.net/qq_39569480/article/details/105249292......
  • 正则表达式基础
    参考:https://blog.csdn.net/weixin_44489823/article/details/100174865,https://blog.csdn.net/m0_62618110/article/details/123704869基础语法"^"指出一个字符串的开始"$"指出一个字符串的结束"\"将下一个字符标记为一个特殊字符、或一个原义字符、或一个向后引用、......
  • Python学习 -- 正则表达式(re模块)
    正则表达式是一种强大的模式匹配工具,用于在文本中查找和匹配特定模式的字符串。在Python中,我们可以使用re模块来操作和处理正则表达式。本篇技术博客将介绍正则表达式的基础语法和re模块的详细使用方法,并通过具体的代码案例来帮助初学者快速掌握正则表达式的使用。正则表达式基础语......
  • 正则表达式使用文档
    通过网站https://regex101.com/可以测试正则表达式的匹配结果及匹配过程.本文章抛开各个编程语言实现差异,仅做正则本身的介绍,会尽量将正则这玩意说明白,使得你看完这边文章后对正则基本可以运用自如.温馨提示,这篇文章会比较长,大致浏览即可.正确的方式是收藏起来,等......