前缀、中缀、后缀表达式及其相互转化的Java实现
(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中的元素并输出,结果的逆序即为中缀表达式对应的后缀表达式(转换为前缀表达式时不用逆序)。
| 扫描到的元素 | 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中剩余的运算符 |
String midToPost(String midSeq){
Stack<Character> S1 = new Stack<Character>();
Stack<Character> S2 = new Stack<Character>();
int len = midSeq.length();
int index = 0;
while(index < len){
char c = midSeq.charAt(index);
switch(c){
case '(':
S1.push(c);
break;
case ')':
while(S1.peek() != '(')
S2.push(S1.pop());
S1.pop();
break;
case '+':
case '-':
while(!S1.empty() && S1.peek() != '(')
S2.push(S1.pop());
S1.push(c);
break;
case '*':
case '/':
while(!S1.empty() && S1.peek().toString().matches("[*/]"))
S2.push(S1.pop());
S1.push(c);
break;
default:
S2.push(c);
}
index++;
}
while(!S1.empty())
S2.push(S1.pop());
Iterator<Character> iter = S2.iterator();
StringBuffer postSeq = new StringBuffer();
while(iter.hasNext())
postSeq.append(iter.next());
return postSeq.toString();
}
String midToPre(String midSeq){
Stack<Character> S1 = new Stack<>(); //S1用来存放临时运算符
Stack<Character> S2 = new Stack<Character>(); //S2用来存放最后结果
int len = midSeq.length();
int index = len - 1;
while(index >= 0){
char c = midSeq.charAt(index);
switch(c){
case ')':
S1.push(c);
break;
case '(':
while(S1.peek() != ')'){
S2.push(S1.pop());
}
S1.pop();
break;
case '*':
case '/':
S1.push(c);
break;
case '+':
case '-':
if(S1.empty() || S1.peek().toString().matches("[+-]"))
S1.push(c);
else{
while(!S1.empty() && S1.peek().toString().matches("[*/]")){
S2.push(S1.pop());
}
S1.push(c);
}
break;
default:
S2.push(c);
}
index--;
}
while(!S1.empty())
S2.push(S1.pop());
StringBuffer preSeq = new StringBuffer();
Iterator<Character> iter = S2.iterator();
while(iter.hasNext())
preSeq.append(iter.next());
preSeq = preSeq.reverse();
return preSeq.toString();
}
前缀、中缀、后缀表达式及其相互转化的Java实现的更多相关文章
- java四则运算----前缀、中缀、后缀表达式
接到一个新需求,需要实现可配置公式,然后按公式实现四则运算. 刚拿到需求,第一反应就是用正则匹配‘(’,‘)’,‘+’,‘-’,‘*’,‘/’,来实现四则运算,感觉不复杂. 然后开始coding.发现 ...
- 前缀、中缀、后缀表达式以及简单计算器的C++实现
前缀表达式(波兰表达式).中缀表达式.后缀表达式(逆波兰表达式) 介绍 三种表达式都是四则运算的表达方式,用以四则运算表达式求值,即数学表达式的求解. 前缀表达式 前缀表达式是一种没有括号的算术表达式 ...
- Atitti. 语法树AST、后缀表达式、DAG、三地址代码
Atitti. 语法树AST.后缀表达式.DAG.三地址代码 抽象语法树的观点认为任何复杂的语句嵌套情况都可以借助于树的形式加以描述.确实,不得不承认应用抽象语法树可以使语句翻译变得相对容易,它很好地 ...
- 深入浅出数据结构C语言版(8)——后缀表达式、栈与四则运算计算器
在深入浅出数据结构(7)的末尾,我们提到了栈可以用于实现计算器,并且我们给出了存储表达式的数据结构(结构体及该结构体组成的数组),如下: //SIZE用于多个场合,如栈的大小.表达式数组的大小 #de ...
- 洛谷P1310 表达式的值 题解 栈/后缀表达式的应用
题目链接:https://www.luogu.org/problem/P1310 本题涉及算法:栈.前缀表达式转后缀表达式,动态规划思想. 这道题目我思考了好长时间,第一时间让我做的话我也做不出来. ...
- Java数据结构和算法(六)——前缀、中缀、后缀表达式
前面我们介绍了三种数据结构,第一种数组主要用作数据存储,但是后面的两种栈和队列我们说主要作为程序功能实现的辅助工具,其中在介绍栈时我们知道栈可以用来做单词逆序,匹配关键字符等等,那它还有别的什么功能吗 ...
- Java数据结构和算法(六):前缀、中缀、后缀表达式
前面我们介绍了三种数据结构,第一种数组主要用作数据存储,但是后面的两种栈和队列我们说主要作为程序功能实现的辅助工具,其中在介绍栈时我们知道栈可以用来做单词逆序,匹配关键字符等等,那它还有别的什么功能吗 ...
- java实现中缀表达式转后缀表达式
package postfix; import java.util.Stack; /** * * @author DELL 将中缀表达式转化为后缀表达式 */ public class Express ...
- 【java】中缀表达式转后缀表达式 java实现
算法: 中缀表达式转后缀表达式的方法:1.遇到操作数:直接输出(添加到后缀表达式中)2.栈为空时,遇到运算符,直接入栈3.遇到左括号:将其入栈4.遇到右括号:执行出栈操作,并将出栈的元素输出,直到弹出 ...
随机推荐
- Unity3D研究院之在MAC上脚本XlsxWriter写入Excel .xlsx格式
原地址:http://www.xuanyusong.com/archives/3011 以前找了很久可以跨平台支持读写Excel的工具,我也试了很多种DLL.可在Windows上各个完美支持,可是在M ...
- MYSQL SHOW VARIABLES简介
原文地址:http://www.2cto.com/database/201108/100546.html mysqld服务器维护两种变量.全局变量影响服务器的全局操作.会话变量影响具体客户端连接相关操 ...
- Win2003部署Framework 4.5框架的MVC4项目
[一篮饭特稀原创,转载请注明出自http://www.cnblogs.com/wanghafan/p/4554672.html] Win2003中IIS6部署Framework 4.5框架的MVC4 ...
- Android UI性能优化详解
设计师,开发人员,需求研究和测试都会影响到一个app最后的UI展示,所有人都很乐于去建议app应该怎么去展示UI.UI也是app和用户打交道的部分,直接对用户形成品牌意识,需要仔细的设计.无论你的ap ...
- Spring事务传播机制
Spring在TransactionDefinition接口中规定了7种类型的事务传播行为,它们规定了事务方法和事务方法发生嵌套调用时事务如何进行传播,即协调已经有事务标识的方法之间的发生调用时的事务 ...
- Emmet(前身是zen coding)介绍和使用
Zen coding - 一种快速编写HTML/CSS代码的方法.它使用仿CSS选择器的语法来快速开发HTML和CSS - 由Sergey Chikuyonok开发. 现在它改名为了Emmet,并且搭 ...
- 如何关闭win7的ps/2兼容鼠标(触屏版)
买了一个新电脑联想ThinkPad E555 可是刚拿到是个win10 的系统,用习惯了win7,win0不太好用, 然后帮我刷成了win7,之后一切都好,性能也是让我很满意,但是却关不掉触控板,于是 ...
- Receving Transactions > No data found IQC无法接收PO采购物料
Receving Transactions > No data found IQC无法接收PO采购物料 APP-PO-14094: No records meet your search cri ...
- 存储过程中“Select Top 变量”的问题如何解决
在SqlServer2005中,可以这样: DECLARE @p int SELECT TOP (@p) * FROM 表名 在SqlServer2000中,不支持以上方法,可以这样: DECLARE ...
- UIButton图片与文字位置调整
1:左图右文 默认效果就行 2:左文右图 UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom]; [btn addTarget:se ...