我的编程空间,编程开发者的网络收藏夹
学习永远不晚

后缀表达式的java如何实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

后缀表达式的java如何实现

这篇“后缀表达式的java如何实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“后缀表达式的java如何实现”文章吧。

中缀表示法java实现

观察一个普通的算式:3+4*5

我们当然知道,应该先计算 4*5 再将这个结果和3相加,就能得到最后的结果。

如果是一个复杂一些的算式:3+4*((5-6)/7+8)

这依然难不倒我们,只要牢记()的优先级最高,然后是*/,最后是+-就没问题了,这就是通常的中缀表示法。

但是通过算法分析,这样的表达式,由于每一次都需要判断优先级,所以运行的时间应当是O(N^2)。

在表达式很长很复杂的时候,就需要一种更适合计算机的算法来计算了。

后缀表示法

简介

逆波兰表示法(Reverse Polish notation,RPN,或逆波兰记法),是一种是由波兰数学家扬·武卡谢维奇1920年引入的数学表达式方式,在逆波兰记法中,所有操作符置于操作数的后面,因此也被称为后缀表示法。

逆波兰记法不需要括号来标识操作符的优先级。逆波兰记法中,操作符置于操作数的后面。

例如表达“三加四”时,写作“3 4 +”,而不是“3 +4”。如果有多个操作符,操作符置于第二个操作数的后面,所以常规中缀记法的“3 - 4 + 5”在逆波兰记法中写作“3 4 - 5+”:先3减去4,再加上5。——维基百科逆波兰表示法词条。

这种表示法有以下特点:

  • 不需要使用括号。和中缀表达式不同,逆波兰表达式不需要遍历整个算式来寻找一对括号。

  • 逆波兰表达式的实现一般基于堆栈。在计算机中,堆栈这种数据结构具有极快的效率。运行时间是O(N)。

  • 不满足交换律。

逆波兰表达式的计算方式

例如2*3+4*5

你可以这么计算,2 和 3 相乘的和是 5,把这个数存起来,再计算 4*5 的值,存起来, 最后在计算两个存在一起的值。写出来是这样子的 2 3 * 4 5 * + 。这就是后缀或逆波兰记法。

采用堆栈实现的过程很简单,规则如下。

从头开始读取。读取到如果是数字,则将其压入栈中。如果是一个符号,就取两次栈顶的元素通过该符号进行计算,再把得到的数压入栈中。

Java实现

public class PRNCalculator {           public static double PRNCal(String PRN){              Stack<Double> stack = new Stack<Double>();              String[] ss = PRN.split(" ");              for(int i = 0; i < ss.length; i++){                     if(ss[i].matches("\\d")) stack.push(Double.valueOf(ss[i]));                     if(ss[i].matches("[+|-|*|/]")){                           double b = stack.pop();                           double a = stack.pop();                           if(ss[i].equals("+")) stack.push(a + b);                           if(ss[i].equals("-")) stack.push(a - b);                           if(ss[i].equals("*")) stack.push(a * b);                           if(ss[i].equals("/")) stack.push(a / b);                     }              }              return stack.pop();       }}

Test类

public class PRNTest {       public static void main(String[] args) {              String s = "2 3 * 4 5 * + ";              double result = PRNCalculator.PRNCal(s);              System.out.println("输入的逆波兰表达式:" + s);              System.out.println("计算结果:" + result);       }}

打印结果:

输入的逆波兰表达式:2 3 * 4 5 * +
计算结果:26.0

与中缀记法的转换

和后缀表达式的计算方法类似,一个中缀记法转换到后缀记法,也可以利用堆栈来实现。

从头开始读取。如果读取到的是数字,将其输出。如果读取到的是符号,则判断该符号的优先级是否高于栈顶或栈为空,是,则压入栈中;否,则将栈顶输出并继续判断。如果读取到的是括号,”(“会直接被压入栈;在读取到”)”的时候,栈会一直弹出直到遇到”(“。下面是这个转换的Java实现。

package PRNCalculator;import java.util.Stack;public class PRNCalculator {       public static String PRNTansf(String s){              Stack<String> stack = new Stack<String>();              String[] ss = s.split(" ");              StringBuffer sb = new StringBuffer();              for(int i = 0; i < ss.length; i++){                     if(ss[i].matches("\\d")) sb.append(ss[i] + " ");                     if(ss[i].matches("[+|-|*|/|(|)]")) {                           if(stack.isEmpty()) {                                  stack.push(ss[i]);                           } else {                                  if(ss[i].matches("[+|-]")) {                                         while(!stack.isEmpty() && !stack.peek().matches("[(]")) sb.append(stack.pop()).append(" ");                                         if(stack.isEmpty() || stack.peek().matches("[(]")) stack.push(ss[i]);                                  }                                  if(ss[i].matches("[*|/]")){                                         while(stack.peek().matches("[*|/]") && !stack.peek().matches("[(]")) sb.append(stack.pop()).append(" ");                                         if(stack.isEmpty() || stack.peek().matches("[(]") || stack.peek().matches("[+|-]")) stack.push(ss[i]);                                  }                                  if(ss[i].matches("[(]")) {                                         stack.push(ss[i]);                                  }                                  if(ss[i].matches("[)]")){                                         while(!stack.peek().matches("[(]")) sb.append(stack.pop()).append(" ");                                         if(stack.peek().matches("[(]")) stack.pop();                                  }                           }                     }              }              while(!stack.isEmpty()) sb.append(stack.pop()).append(" ");              return sb.toString();       }}
* Test类*package PRNCalculator;public class PRNTest {       public static void main(String[] args) {              String s = "4 + 5 + 8 * ( 6 + 8 * 7 ) / 3 + 4";              String PRN = PRNCalculator.PRNTansf(s);              System.out.println("输入的表达式为:");              System.out.println(s);              System.out.println("输出的逆波兰表达式为:");              System.out.println(PRN);              double result = PRNCalculator.PRNCal(PRN);              System.out.println("该表达式计算结果为:");              System.out.println(result);       }}

打印结果:

输入的表达式为:
4 + 5 + 8 * ( 6 + 8 * 7 ) / 3 + 4
输出的逆波兰表达式为:
4 5 + 8 6 8 7 * + * 3 / + 4 +
该表达式计算结果为:
178.33333333333334

java后缀表达式的计算

实现方法

从左至右扫描表达式

遇到数字时,将数字压栈,遇到运算符时,弹出栈顶的两个数,计算并将结果入栈

重复2直到表达式最右端,最后运算得出的值即为表达式的结果

示例

计算后缀表达式的值:1 2 3 + 4 &times; + 5 -

从左至右扫描,将1,2,3压入栈;

遇到+运算符,3和2弹出,计算2+3的值,得到5,将5压入栈;

遇到4,将4压入栈

遇到&times;运算符,弹出4和5,计算5&times;4的值,得到20,将20压入栈;

遇到+运算符,弹出20和1,计算1+20的值,得到21,将21压入栈;

遇到5,将5压入栈;

遇到-运算符,弹出5和21,计算21-5的值,得到16为最终结果

代码实现

public class ReversePolishNotation {    public static void main(String[] args) {        String notation = "10 2 3 + 4 * + 5 -";        ReversePolishNotation reversePN = new ReversePolishNotation();        Stack<Integer> numStack = new Stack<>();        //以空格分隔上述表达式,存到数组中        String[] s = notation.split(" ");        //遍历数组        for (int i = 0; i < s.length; i++) {            if (!reversePN.isOperator(s[i])){                //如果不是运算符,则压栈                numStack.push(Integer.parseInt(s[i]));            } else {                //为运算符,则取出栈顶的两个数字进行运算                int result = reversePN.calculation(numStack.pop(), numStack.pop(), s[i]);                //将结果压栈                numStack.push(result);            }        }        //循环结束,栈中仅剩的一个元素及为结果        System.out.println(numStack.pop());    }    //判断是否是运算符    public boolean isOperator(String oper){        return oper.equals("+") ||oper.equals("-")  ||oper.equals("*")  ||oper.equals("/") ;    }    //计算    public int calculation(int num1, int num2, String oper){        switch (oper){            case "+":                return num2 + num1;            case "-":                return num2 - num1;            case "*":                return num2 * num1;            case "/":                return num2 / num1;            default:                return 0;        }    }}

以上就是关于“后缀表达式的java如何实现”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

后缀表达式的java如何实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

后缀表达式的java如何实现

这篇“后缀表达式的java如何实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“后缀表达式的java如何实现”文章吧。中缀表
2023-07-02

C++如何实现中缀表达式转化为后缀表达式

这篇文章将为大家详细讲解有关C++如何实现中缀表达式转化为后缀表达式,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。1.题目描述所谓后缀表达式是指这样的一个表达式:式中不再引用括号,运算符号放在两个运算对象
2023-06-29

C++ 如何使用栈求解中缀、后缀表达式的值

这篇文章主要介绍了C++ 使用栈求解中缀、后缀表达式的值,本文讲解了中缀、后缀表达式的求值过程以及如何将一个中缀表达式转换成后缀表达式,需要的朋友可以参考下
2022-11-13

Java数据结构和算法之前缀、中缀和后缀表达式的示例分析

小编给大家分享一下Java数据结构和算法之前缀、中缀和后缀表达式的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!1、人如何解析算术表达式如何解析算术表达式?或者换种说法,遇到某个算术表达式,我们是如何计算的:①、求
2023-06-28

JAVA中如何实现表达式计算源码

这篇文章主要为大家展示了“JAVA中如何实现表达式计算源码”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“JAVA中如何实现表达式计算源码”这篇文章吧。支持运算符:+-*/%><][!|&=;其中
2023-06-03

怎么在python中利用后缀表达式实现一个计算器功能

本文章向大家介绍怎么在python中利用后缀表达式实现一个计算器功能的基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。前缀表达式运算符在数字的前面1 + (2 + 3) * 4 - 5 (中缀)- + 1 * +
2023-06-06

python如何实现三元表达式

这篇文章主要介绍了python如何实现三元表达式,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。三元表达在许多用例中,我们需要根据条件定义具有特定值的变量,并且我们可以简单地使
2023-06-27

LINQ如何实现查询表达式

这篇文章主要介绍了LINQ如何实现查询表达式,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。新建一个叫Step1.aspx的新页面。添加一个GridView控件到页面中,如下所
2023-06-17

java如何提取表达式

这篇文章主要为大家展示了“java如何提取表达式”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“java如何提取表达式”这篇文章吧。提取表达式在软件开发过程中,程序员很容易有意无意让代码做一些“重
2023-06-27

python如何实现列表推导表达式

这篇文章主要为大家展示了“python如何实现列表推导表达式”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“python如何实现列表推导表达式”这篇文章吧。列表推导表达式列表推导表达式是创建列表的
2023-06-27

java如何利用JEXL实现动态表达式编译

小编给大家分享一下java如何利用JEXL实现动态表达式编译,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!背景做项目突然遇到这样的需求:系统要获取多个数据源的数据
2023-06-14

Java的Lambda表达式如何使用

这篇文章主要介绍“Java的Lambda表达式如何使用”,在日常操作中,相信很多人在Java的Lambda表达式如何使用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的Lambda表达式如何使用”的疑
2023-06-30

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录