Study

(JAVA) JAVA Stack을 활용하여 infix-postfix 변환 및 계산

Developer RyanKim 2019. 3. 15. 16:59
//1. 수식에서 첫번째 항목을 빼냅니다.

//2. 수식의 빼낸 항목이 피연산자라면 그대로 출력측으로 빼냅니다

//3. 수식의 빼낸 항목이 연산자라면 스택의 최상위 원소와 비교합니다.

//4. 빼낸 연산자의 우선순위가 스택의 최상위에 있는 원소보다 높으면 그대로 스택에 넣습니다.

//5. 그렇지 않다면 스택의 최상위 원소를 출력측으로 뺴내고 빼낸 연산자를 스택에 넣습니다.

//6. 단, 여는 괄호'('가 나오면 닫는괄호')'가 나올 때까지 스택에 남아있게 되며, 닫는괄호')'가 나타나면

그때까지 여는괄호'('보다 상위에 있는 모든 연산자를 출력측으로 빼냅니다.


정말 열심히 찾다가 어디선가 푸는 방식을 주석으로 코드에 넣은건데 출처를 잊었네요..

 이 방법대로 구현하면 infix -> postfix 변환이 잘됩니다.


public class InfixToPostfix {
private Stack stack;
public String infix = "";
public String inputToPostfix = "";

public InfixToPostfix() {
this.stack = new Stack<>();
}

public Stack getStack() {
return stack;
}

public int calcPostfix(){
for(int i = 0; i < inputToPostfix.length(); i++){
char token = inputToPostfix.charAt(i);
System.out.println("clac token : " + token);

if(isNumber(token)) stack.push(Character.digit(token,10));
else{
int n1 = (Integer) stack.pop();
int n2 = (Integer) stack.pop();
int operateRes = operateToken(token,n2,n1);

System.out.println("operateRes : " + operateRes);
stack.push(operateRes);
}
System.out.println(stack);
}
return (Integer)stack.pop();
}

private int operateToken(char operater,int n1, int n2){
switch (operater){
case '*' :
return n1 * n2;
case '/' :
return n1 / n2;
case '+' :
return n1 + n2;
case '-' :
return n1 - n2;
}
return 0;
}

public String getOutput(){
pushInStackForPostfix();
return inputToPostfix;
}
private void pushInStackForPostfix(){
for(int i=0; i < infix.length(); i++){
char token = infix.charAt(i);
System.out.println("token : " + token);

if(isNumber(token)){
inputToPostfix += token;
}else{
if(token == '('){
stack.push(token);
}else{
sortOperater(token);
}

System.out.println("==out==\n" + inputToPostfix);
}
}
clearStack();
}

private void sortOperater(char input){
System.out.println("==stack==\n" + stack.toString() + "\n ====");
if(stack.empty()){ stack.push(input); return;}

if(input == ')'){
while(!stack.isEmpty()){
char inClose = (Character) stack.pop();
if(inClose == '(') break;
inputToPostfix += inClose;
}
}else {
int inputGarde = getOperaterGrade(input);

while (!stack.isEmpty()) {
char prev = (Character)stack.pop();
if(prev == '('){
stack.push(prev);
break;
}
int prevGarde = getOperaterGrade(prev);

if (prevGarde >= inputGarde) {
inputToPostfix += prev;
} else {
stack.push(prev);
break;
}

}
stack.push(input);
}
}

public boolean isNumber(char token){
if('0' <= token && token <= '9')
return true;
else
return false;
}

private int getOperaterGrade(char operater){
int grade = 0;
switch (operater){
case '*' : case '/' : case '%':
grade = 2;
break;
case '+' : case '-' :
grade = 1;
break;
case '(' :
grade = 0;
break;
}
return grade;
}

private void clearStack(){
char token;
while(!stack.empty()){
token = (Character)stack.pop();
inputToPostfix += token;
}
}
}


작성하다가 힘이 빠져서 코드가 정리가 안된 부분도 많이 있을것 같습니다. 잘 걸러서 참고해주세요!


문제 풀면서 막혔던 부분은 

1. if else 조건을 줄 때 무슨조건을 먼저 검사할 것인지

2. Stack에 담을 객체의 타입 - 코드를 보시면 처음에 스택 하나로 postfix로 변환시에는 Char형을 넣어 사용하고 그후 계산을 할때는 Integer 형을 사용하기 때문에 두개를 변환하는 과정 등을 신경써야 했습니다.

3. while문의 종료 조건 - 반복문의 종료조건을 잘 확인하고 걸어야 했습니다.

4. 메소드를 나눌 단위 - 이 부분이 제일 어려웠습니다. 결국 나누다가 포기했지요... 잘게 쪼갤 수록 오히려 조건문이 더 많아지고 복잡해지는 느낌이 들었습니다.