前言:一个表达式有中缀表达式,前缀表达式,后缀表达式三种表示方法,
如1+2*3-8/2=3,因为操作符以中缀形式处于操作数的中间叫做中缀表达式
后缀表达式:运算符在操作数的后面。 上式的后缀表达式就是123*+82/-
虽然人的大脑很容易理解与分析中缀表达式,但对计算机来说中缀表达式是很复杂的,因此计算表达式的值时,通常需要先将中缀表达式转换为前缀或后缀表达式,然后再进行求值。对计算机来说,计算前缀或后缀表达式的值非常简单。
中缀表达式转后缀
要将一个中缀表达式转化成后缀表达式,需要用到栈这个数据结构,它有一个重要的性质:后进先出,这是这个程序的基础。
步骤:
初始化一个运算符栈,用于存储运算符。 初始化一个list,用于存放后缀表达式,然后从左到右扫描表达式
遇到数字,直接存到list中
当读到运算符时,分两种情况讨论
当运算符栈为空或者栈顶运算符的优先级小于当前运算符的优先级时,或者栈顶为左括号时,直接入栈。
运算符栈不为空且栈顶操作符的优先级大于当前运算符优先级时,循环执行运算符出栈并加入list中,直到遇到优先级小于等于当前运算符的元素为止,循环执行完后再将当前运算符压栈
当遇到左括号时,直接压栈
当遇到右括号时,循环执行出栈操作并加入到list中,直到遇到左括号为止,并将左括号弹出,但不加入到list中,丢弃这对括号
实现代码
import java.util.ArrayDeque;
import java.util.Deque;
import java.util.Iterator;
import java.util.Scanner;
import java.util.ArrayList;
/**
* date :2020-12-2
* author: wingchi-ye
* 测试: 输入表达式正常时运行正常
* 没有校验表达式格式正确性
*/
class test{
public static void main(String args[]) throws IOException {
char str[];
Scanner Reader =new Scanner(System.in) ;
Calculator cal =new Calculator();
for(int i=0;i<2;i++){
str =Reader.nextLine().toCharArray();
ArrayList list = cal.toReverse_polish_type(str); /*转换成逆波兰式 */
cal.calaExp(list) ; /**计算逆波兰式 */
}
Reader.nextLine();
}
}
class Calculator {
public void calaExp(ArrayList exp){
Iterator it=exp.iterator();
Deque stack =new ArrayDeque();
while(it.hasNext()) {
char ch =it.next() ;
if(ch-'0' <=9 &&ch-'0'>=0) stack.addFirst((int)ch-'0') ;
else {
stack.addFirst( calc (stack.removeFirst() , stack.removeFirst() ,ch )) ;
}
}
int ans= stack.removeFirst();
System.out.println(ans);
}
public int calc(int a1,int a2,char ch) {
switch(ch){
case 'x' : return a1*a2;
case '/': return a2/a1;
case '+': return a1+a2;
case '-': return a2-a1 ;
case '*': return a1*a2;
default : return 0;
}
}
public ArrayList toReverse_polish_type (char[] str){
Deque stack =new ArrayDeque();
ArrayList expression =new ArrayList<>() ;
Character ch ;
for(int i=0;i
ch = str[i] ;
if(ch-'0' <= 9 && ch-'0' >=0 )
expression.add(ch) ;
else if(ch=='(' || ch==')'){
if(ch=='(') stack.addFirst(ch) ;
else {
/**右括号:依次弹出运算符栈的运算符并压入list,直到遇到左括号为止,然后将这对括号丢弃 */
char tmp = ' ';
while( (tmp=stack.removeFirst()) !='(') {
expression.add(tmp) ;
}
}
}
else {
if(stack.isEmpty() || isVaild(ch,stack.getFirst()))
stack.addFirst(ch) ;
else {
/**优先级小于等于的*/
while(stack.isEmpty()==false && !isVaild(ch,stack.peekFirst()) ){
expression.add(stack.removeFirst());
}
stack.addFirst(ch) ; /** */
}
}
}
while(!stack.isEmpty()) expression.add(stack.removeFirst()) ;
showlist(expression) ;
return expression;
}
/**
* 比较两个运算符的优先级,并且判断栈顶元素是否为左括号
* @param s1
* @param inStack:栈顶元素
* @return s1优先级和s2高 : true
* 相等或小于返回false
*/
public boolean isVaild(char s1,char inStack){
if(inStack=='(') return true ;
if(s1=='x' || s1=='/' || s1=='*') {
if(inStack=='+' || inStack=='-') return true ;
return false ;
}
return false;
}
public void showlist(ArrayList list){
for(int i=0;i
System.out.print(list.get(i)) ;
}
System.out.println() ;
}
}
标签:ch,Java,实现,list,运算符,波兰,return,stack,表达式
来源: https://blog.csdn.net/zzzlueng/article/details/110500899