一、中缀表达式转换为前缀、后缀表达式
给个中缀表达式:a+b*c-(d+e)
   首先根据运算符的优先级给所有运算单位加括号:((a+(b*c))-(d+e))
   将运算符号移动到对应括号的前面然后去掉所有括号就转换为前缀表达式:
     -( +(a *(bc)) +(de)) ->  -+a*bc+de
   将运算符号移动到对应括号的后面然后去掉所有括号就转换为后缀表达式:
     ((a(bc)* )+ (de)+ )-  ->   abc*+de+-
二、前缀表达式和后缀表达式的计算
   前缀表达式的计算:从右至左扫描表达式,遇到数字的时候将数字入栈,遇到运算符时,弹出栈顶的两个数,用运算符对它们做相应运算(栈顶元素op次顶元素),并将结果入栈,重复该操作直到表达式最左端,最后算出的值即为表达式的结果。
   后缀表达式的计算:从左至右扫描表达式,遇到数字时,将数字压入堆栈,遇到运算符时,弹出栈顶的两个数,用运算符对它们做相应的计算(次顶元素 op 栈顶元素),并将结果入栈;重复上述过程直到表达式最右端,最后运算得出的值即为表达式的结果。
   注意:这里都是前面元素op后面元素,但是因为前缀表达式在计算的时候是后面元素先入栈,所以是栈顶元素op次顶元素;后缀表达式是前面元素先入栈,所以是次顶元素op栈顶元素。


三、将中缀表达式转换为前缀表达式
遵循以下步骤:

(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中的元素并输出,结果即为中缀表达式对应的前缀表达式。

例如,将中缀表达式“1+((2+3)×4)-5”转换为前缀表达式的过程如下:

扫描到的元素 S2(栈底->栈顶) S1 (栈底->栈顶) 说明
5 5 数字,直接入栈
- 5 - S1为空,运算符直接入栈
) 5 - ) 右括号直接入栈
4 5 4 - ) 数字直接入栈
× 5 4 - ) × S1栈顶是右括号,直接入栈
) 5 4 - ) × ) 右括号直接入栈
3 5 4 3 - ) × ) 数字
+ 5 4 3 - ) × ) + S1栈顶是右括号,直接入栈
2 5 4 3 2 - ) × ) + 数字
( 5 4 3 2 + - ) × 左括号,弹出运算符直至遇到右括号
( 5 4 3 2 + × - 同上
+ 5 4 3 2 + × - + 优先级与-相同,入栈
1 5 4 3 2 + × 1 - + 数字
到达最左端 5 4 3 2 + × 1 + - S1中剩余的运算符

因此结果为“- + 1 × + 2 3 4 5”。


四、后缀表达式转换为中缀表达式
与转换为前缀表达式相似,遵循以下步骤:

(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中的元素并输出,结果的逆序即为中缀表达式对应的后缀表达式(转换为前缀表达式时不用逆序)。


例如,将中缀表达式“1+((2+3)×4)-5”转换为后缀表达式的过程如下:

扫描到的元素 S2(栈底->栈顶) S1 (栈底->栈顶) 说明
1 1 数字,直接入栈
+ 1 + S1为空,运算符直接入栈
( 1 + ( 左括号,直接入栈
( 1 + ( ( 同上
2 1 2 + ( ( 数字
+ 1 2 + ( ( + S1栈顶为左括号,运算符直接入栈
3 1 2 3 + ( ( + 数字
) 1 2 3 + + ( 右括号,弹出运算符直至遇到左括号
× 1 2 3 + + ( × S1栈顶为左括号,运算符直接入栈
4 1 2 3 + 4 + ( × 数字
) 1 2 3 + 4 × + 右括号,弹出运算符直至遇到左括号
- 1 2 3 + 4 × + - -与+优先级相同,因此弹出+,再压入-
5 1 2 3 + 4 × + 5 - 数字
到达最右端 1 2 3 + 4 × + 5 - S1中剩余的运算符

因此结果为“1 2 3 + 4 × + 5 -”(注意需要逆序输出)。


五、编程实现
  1. String midToPost(String midSeq){
  2. Stack<Character> S1 = new Stack<Character>();
  3. Stack<Character> S2 = new Stack<Character>();
  4.  
  5. int len = midSeq.length();
  6. int index = 0;
  7. while(index < len){
  8. char c = midSeq.charAt(index);
  9. switch(c){
  10. case '(':
  11. S1.push(c);
  12. break;
  13. case ')':
  14. while(S1.peek() != '(')
  15. S2.push(S1.pop());
  16. S1.pop();
  17. break;
  18. case '+':
  19. case '-':
  20. while(!S1.empty() && S1.peek() != '(')
  21. S2.push(S1.pop());
  22. S1.push(c);
  23. break;
  24. case '*':
  25. case '/':
  26. while(!S1.empty() && S1.peek().toString().matches("[*/]"))
  27. S2.push(S1.pop());
  28. S1.push(c);
  29. break;
  30. default:
  31. S2.push(c);
  32. }
  33. index++;
  34. }
  35. while(!S1.empty())
  36. S2.push(S1.pop());
  37. Iterator<Character> iter = S2.iterator();
  38. StringBuffer postSeq = new StringBuffer();
  39. while(iter.hasNext())
  40. postSeq.append(iter.next());
  41. return postSeq.toString();
  42. }
  43.  
  44. String midToPre(String midSeq){
  45. Stack<Character> S1 = new Stack<>(); //S1用来存放临时运算符
  46. Stack<Character> S2 = new Stack<Character>(); //S2用来存放最后结果
  47.  
  48. int len = midSeq.length();
  49. int index = len - 1;
  50. while(index >= 0){
  51. char c = midSeq.charAt(index);
  52. switch(c){
  53. case ')':
  54. S1.push(c);
  55. break;
  56. case '(':
  57. while(S1.peek() != ')'){
  58. S2.push(S1.pop());
  59. }
  60. S1.pop();
  61. break;
  62. case '*':
  63. case '/':
  64. S1.push(c);
  65. break;
  66. case '+':
  67. case '-':
  68. if(S1.empty() || S1.peek().toString().matches("[+-]"))
  69. S1.push(c);
  70. else{
  71. while(!S1.empty() && S1.peek().toString().matches("[*/]")){
  72. S2.push(S1.pop());
  73. }
  74. S1.push(c);
  75. }
  76. break;
  77. default:
  78. S2.push(c);
  79. }
  80. index--;
  81. }
  82. while(!S1.empty())
  83. S2.push(S1.pop());
  84. StringBuffer preSeq = new StringBuffer();
  85. Iterator<Character> iter = S2.iterator();
  86. while(iter.hasNext())
  87. preSeq.append(iter.next());
  88. preSeq = preSeq.reverse();
  89. return preSeq.toString();
  90. }

参考资料:

前缀、中缀、后缀表达式及其相互转化的Java实现的更多相关文章

  1. java四则运算----前缀、中缀、后缀表达式

    接到一个新需求,需要实现可配置公式,然后按公式实现四则运算. 刚拿到需求,第一反应就是用正则匹配‘(’,‘)’,‘+’,‘-’,‘*’,‘/’,来实现四则运算,感觉不复杂. 然后开始coding.发现 ...

  2. 前缀、中缀、后缀表达式以及简单计算器的C++实现

    前缀表达式(波兰表达式).中缀表达式.后缀表达式(逆波兰表达式) 介绍 三种表达式都是四则运算的表达方式,用以四则运算表达式求值,即数学表达式的求解. 前缀表达式 前缀表达式是一种没有括号的算术表达式 ...

  3. Atitti. 语法树AST、后缀表达式、DAG、三地址代码

    Atitti. 语法树AST.后缀表达式.DAG.三地址代码 抽象语法树的观点认为任何复杂的语句嵌套情况都可以借助于树的形式加以描述.确实,不得不承认应用抽象语法树可以使语句翻译变得相对容易,它很好地 ...

  4. 深入浅出数据结构C语言版(8)——后缀表达式、栈与四则运算计算器

    在深入浅出数据结构(7)的末尾,我们提到了栈可以用于实现计算器,并且我们给出了存储表达式的数据结构(结构体及该结构体组成的数组),如下: //SIZE用于多个场合,如栈的大小.表达式数组的大小 #de ...

  5. 洛谷P1310 表达式的值 题解 栈/后缀表达式的应用

    题目链接:https://www.luogu.org/problem/P1310 本题涉及算法:栈.前缀表达式转后缀表达式,动态规划思想. 这道题目我思考了好长时间,第一时间让我做的话我也做不出来. ...

  6. Java数据结构和算法(六)——前缀、中缀、后缀表达式

    前面我们介绍了三种数据结构,第一种数组主要用作数据存储,但是后面的两种栈和队列我们说主要作为程序功能实现的辅助工具,其中在介绍栈时我们知道栈可以用来做单词逆序,匹配关键字符等等,那它还有别的什么功能吗 ...

  7. Java数据结构和算法(六):前缀、中缀、后缀表达式

    前面我们介绍了三种数据结构,第一种数组主要用作数据存储,但是后面的两种栈和队列我们说主要作为程序功能实现的辅助工具,其中在介绍栈时我们知道栈可以用来做单词逆序,匹配关键字符等等,那它还有别的什么功能吗 ...

  8. java实现中缀表达式转后缀表达式

    package postfix; import java.util.Stack; /** * * @author DELL 将中缀表达式转化为后缀表达式 */ public class Express ...

  9. 【java】中缀表达式转后缀表达式 java实现

    算法: 中缀表达式转后缀表达式的方法:1.遇到操作数:直接输出(添加到后缀表达式中)2.栈为空时,遇到运算符,直接入栈3.遇到左括号:将其入栈4.遇到右括号:执行出栈操作,并将出栈的元素输出,直到弹出 ...

随机推荐

  1. 【形式化方法:VDM++系列】1.前言

    1.前言 今天开始上课学习软件需求分析与VDM++,经过一节课的学习,我又增长了见识. 软件需求工程在软件工程中处于十分核心的地位:需求分析的好坏直接决定软件工程的成败.这一点是我之前对需求工程的理解 ...

  2. 安装java memcached client到本地maven repository

    由于目前java memcached client没有官方的maven repository可供使用,因此使用时需要手动将其安装到本地repository.java memcached client的 ...

  3. SPRING IN ACTION 第4版笔记-第三章ADVANCING WIRING-007-给BEAN运行时注入值placeholder、@Value

    一.用placeholder给bean运行时注入值的步骤 Spring取得placeholder的值是用${...} 1.声明placeholder bean (1)java方式 In order t ...

  4. Java按位置解析文本文件(使用Swing选择文件)

    工作中遇到这样的一个需求,按位置解析一些文本文件,它们由头部.详情.尾部组成,并且每一行的长度可能不一样,每一行代表的意思也可能不一样,但是每一行各个位置代表的含义已经确定了. 例如有下面这样一段文本 ...

  5. Linux内核中流量控制

    linux内核中提供了流量控制的相关处理功能,相关代码在net/sched目录下:而应用层上的控制是通过iproute2软件包中的tc来实现, tc和sched的关系就好象iptables和netfi ...

  6. 加载网络映射盘中的assembly失败的处理办法

    错误症状: 1.{"未能加载文件或程序集“file://*****”或它的某一个依赖项.不支持操作. (异常来自 HRESULT:0x80131515)":"file:/ ...

  7. ArrayList和LinkedList

    ArrayListArrayList其实是包装了一个数组 Object[],当实例化一个ArrayList时,一个数组也被实例化,当向ArrayList中添加对象是,数组的大小也相应的改变.这样就带来 ...

  8. C#中的几个线程同步对象方法

    在编写多线程程序时无可避免会遇到线程的同步问题.什么是线程的同步呢? 举个例子:如果在一个公司里面有一个变量记录某人T的工资count=100,有两个主管A和B(即工作线程)在早一些时候拿了这个变量的 ...

  9. iOS开发 .framework的Optional(弱引用)和Required(强引用)区别, 有错误 Library not found………………

    http://www.cnblogs.com/wanyakun/p/3494323.html 强引用(Required)的framework是一定会被加载到内存的,但是弱引用(Optional)的fr ...

  10. golang安装卸载 linux+windows+raspberryPI 平台

    参考  https://golang.org/doc/install 自ECUG2013洗脑回来,就渴望早点接触Go 听着许式伟和谢孟军的演讲 发现go的网络库的确很强大,高负载利器,语言的一些精简导 ...