스텍을 이용하여 중위연산에서 후위연산으로 변환하는...(이해를 잘 못하겟네요 ;)

leekukheon의 이미지

복학 준비하면서 공부를 하고 있는데요 c로쓴 자료구조라는 책을 보면서 공부하고 있습니다
그런데 스텍 중위연산 후위연산 개념은 이해가 되는데 소스가 잘 이해가 안되네요 ㅠㅠ
여유가 괜찮으시다면 보충설명(주석) 좀 부탁드릴게요....
공부하다가 막히니 흐름도 끊켜서 인터넷하게되네요 ㅠ
=============================================================================

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
 
typedef enum { lparen, rparen, plus, minus, times, divide, 
							mod, eos, operand } precedence ;
 
typedef precedence element;
 
#define MAX_STACK_SIZE 100	/* 최대 스택 크기 */
precedence stack[MAX_STACK_SIZE];
int top = -1;
 
/*	isp와 icp 배열 -- 인덱스는 연산자 lparen, rparen,
	plus, minus, times, divide, mod, eos의 우선순위 값 */
static int isp[] = {  0, 19, 12, 12, 13, 13, 13, 0 };
static int icp[] = { 20, 19, 12, 12, 13, 13, 13, 0 };
char expr[100];
 
precedence get_token (char *symbol, int * n);
 
void postfix(void);
 
int eval(void);
 
int main(void)
{
	printf("Enter The Infix String : ");
	scanf("%s", expr);
	postfix();
	printf ("Postfix String :%d\n", eval());
	return 0;
}
 
void addS(int *top, element item);
element deleteS(int *top);
 
int eval(void)
{	/* 전역 변수로 되어 있는 후위 표기식 expr을 연산한다 */
	precedence token;
	char symbol;
	int op1, op2;
	int n = 0;	/* 수식 문자열을 위한 카운터 */
	int top = -1;
	token = get_token(&symbol, &n);
	while (token != eos) {
		if (token == operand)
			addS(&top, symbol - '0');	/* 스택 삽입 */
		else {	
			/* 두 피연산자를 삭제하여 연산을 수행한 후, 그 결과를 스택에 삽입함 */
			op2 = deleteS(&top);
			op1 = deleteS(&top);	/* 스택 삭제 */
			switch (token) {
				case plus:	addS(&top, op1 + op2); break;
				case minus:	addS(&top, op1 - op2); break;
				case times:	addS(&top, op1 * op2); break;
				case divide:addS(&top, op1 / op2); break;
				case mod:	addS(&top, op1 % op2); break;
			}			
		}
		token = get_token(&symbol, &n);
	}
	return deleteS(&top);
}
 
precedence get_token (char *symbol, int * n)
{
/* 다음 토큰을 취한다. symbol은 문자 표현이며, 
    token은 그것의 열거된 값으로 표현되고, 명칭으로 반환된다. */
	*symbol = expr[(*n)++];
	// printf("%c", *symbol);
	switch (*symbol) {
		case '('	: return lparen;
		case ')' 	: return rparen;
		case '+'	: return plus;
		case '-'	: return minus;
		case '/'	: return divide;
		case '*'	: return times;
		case '%'	: return mod;
		case '\0'	: return eos;
		default		: return operand;
   }
}
 
 
// 변환 부분
void print_token(precedence token);
char post_expr[100];
void postfix(void)
{
	char symbol; 
	precedence token; 
	int n = 0;
	int top = 0; 
	stack[0] = eos; /* eos를 스택에 놓는다 */
	for (token = get_token(&symbol, &n); token != eos; token = get_token(&symbol, &n)) {
		//putchar(symbol);
		if (token == operand)
			print_token(symbol);
		else if (token == rparen) { 
			/* 왼쪽 괄호가 나올 때까지 토큰들을 제거해서 출력 */
			while (stack[top] != lparen)
				print_token(deleteS(&top));
			deleteS(&top);		/* 좌괄호를 버린다 */
		}
		else {
			/* symbol의 isp가 token의 icp보다 크거나 같으면 symbol을 제거하고 출력 */
			while (isp[stack[top]] >= icp[token])
				print_token(deleteS(&top));
			addS(&top, token);
		}
	}
	while ((token = deleteS(&top)) != eos)
		print_token(token);
	print_token(eos);
 
	strcpy(expr, post_expr);
	puts(expr);
}
 
void print_token(precedence token)
{
	static int index = 0;
	char aChar;
 
	switch (token) {
		case plus	: aChar = '+'; break;
		case minus	: aChar = '-'; break;
		case divide	: aChar = '/'; break;
		case times	: aChar = '*'; break;
		case mod	: aChar = '%'; break;
		case eos	: aChar = '\0'; break;
		default		: aChar = token;
	}
 
	post_expr[index++] = aChar;
}
 
void stack_full();
void addS(int *top, element item)
{
	/* 전역 스택에 item을 삽입 */
	if (*top >= MAX_STACK_SIZE - 1) {
		stack_full();
		return;
	}
	stack[++*top] = item;
}
 
element stack_empty();
element deleteS(int *top)
{
	/* stack의 최상위 원소를 반환 */
	if (*top == -1)
		return stack_empty();
	return stack[(*top)--];
}
 
void stack_full()
{
	exit (1);
}
 
element stack_empty()
{
	exit (1);
}

=================================================================================
simpid의 이미지


1) 아래와 같이 코드 태그를 이용하시면 가독성이 좋아집니다.

2) 요청하신 주석은 달아놓지 않았습니다만(원본 그대로 다시올린겁니다.)
이미 주석은 대충 달려 있는것 같습니다.
어느부분이 이해가 안되는건지 구체적으로 얘기해 주시면 여러분들이 나서서 도움을 주실껍니다.

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
 
typedef enum { lparen, rparen, plus, minus, times, divide,
mod, eos, operand } precedence ;
 
typedef precedence element;
 
#define MAX_STACK_SIZE 100 /* 최대 스택 크기 */
precedence stack[MAX_STACK_SIZE];
int top = -1;
 
/* isp와 icp 배열 -- 인덱스는 연산자 lparen, rparen,
plus, minus, times, divide, mod, eos의 우선순위 값 */
static int isp[] = {  0, 19, 12, 12, 13, 13, 13, 0 };
static int icp[] = { 20, 19, 12, 12, 13, 13, 13, 0 };
char expr[100];
 
precedence get_token (char *symbol, int * n);
 
void postfix(void);
 
int eval(void);
 
int main(void)
{
	printf("Enter The Infix String : ");
	scanf("%s", expr);
	postfix();
	printf ("Postfix String :%d\n", eval());
	return 0;
}
 
void addS(int *top, element item);
element deleteS(int *top);
 
int eval(void)
{ /* 전역 변수로 되어 있는 후위 표기식 expr을 연산한다 */
	precedence token;
	char symbol;
	int op1, op2;
	int n = 0; /* 수식 문자열을 위한 카운터 */
	int top = -1;
	token = get_token(&symbol, &n);
	while (token != eos) {
		if (token == operand)
			addS(&top, symbol - '0'); /* 스택 삽입 */
		else {
			/* 두 피연산자를 삭제하여 연산을 수행한 후, 그 결과를 스택에 삽입함 */
			op2 = deleteS(&top);
			op1 = deleteS(&top); /* 스택 삭제 */
			switch (token) {
case plus: addS(&top, op1 + op2); break;
case minus: addS(&top, op1 - op2); break;
case times: addS(&top, op1 * op2); break;
case divide:addS(&top, op1 / op2); break;
case mod: addS(&top, op1 % op2); break;
			}
		}
		token = get_token(&symbol, &n);
	}
	return deleteS(&top);
}
 
precedence get_token (char *symbol, int * n)
{
	/* 다음 토큰을 취한다. symbol은 문자 표현이며,
	token은 그것의 열거된 값으로 표현되고, 명칭으로 반환된다. */
	*symbol = expr[(*n)++];
	// printf("%c", *symbol);
	switch (*symbol) {
case '(' : return lparen;
case ')' : return rparen;
case '+' : return plus;
case '-' : return minus;
case '/' : return divide;
case '*' : return times;
case '%' : return mod;
case '\0' : return eos;
default : return operand;
	}
}
 
 
// 변환 부분
void print_token(precedence token);
char post_expr[100];
void postfix(void)
{
	char symbol;
	precedence token;
	int n = 0;
	int top = 0;
	stack[0] = eos; /* eos를 스택에 놓는다 */
	for (token = get_token(&symbol, &n); token != eos; token = get_token(&symbol, &n)) {
		//putchar(symbol);
		if (token == operand)
			print_token(symbol);
		else if (token == rparen) {
			/* 왼쪽 괄호가 나올 때까지 토큰들을 제거해서 출력 */
			while (stack[top] != lparen)
				print_token(deleteS(&top));
			deleteS(&top); /* 좌괄호를 버린다 */
		}
		else {
			/* symbol의 isp가 token의 icp보다 크거나 같으면 symbol을 제거하고 출력 */
			while (isp[stack[top]] >= icp[token])
				print_token(deleteS(&top));
			addS(&top, token);
		}
	}
	while ((token = deleteS(&top)) != eos)
		print_token(token);
	print_token(eos);
 
	strcpy(expr, post_expr);
	puts(expr);
}
 
void print_token(precedence token)
{
	static int index = 0;
	char aChar;
 
	switch (token) {
case plus : aChar = '+'; break;
case minus : aChar = '-'; break;
case divide : aChar = '/'; break;
case times : aChar = '*'; break;
case mod : aChar = '%'; break;
case eos : aChar = '\0'; break;
default : aChar = token;
	}
 
	post_expr[index++] = aChar;
}
 
void stack_full();
void addS(int *top, element item)
{
	/* 전역 스택에 item을 삽입 */
	if (*top >= MAX_STACK_SIZE - 1) {
		stack_full();
		return;
	}
	stack[++*top] = item;
}
 
element stack_empty();
element deleteS(int *top)
{
	/* stack의 최상위 원소를 반환 */
	if (*top == -1)
		return stack_empty();
	return stack[(*top)--];
}
 
void stack_full()
{
	exit (1);
}
 
element stack_empty()
{
	exit (1);
}

댓글 달기

Filtered HTML

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

BBCode

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param>
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

Textile

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • You can use Textile markup to format text.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Markdown

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • Quick Tips:
    • Two or more spaces at a line's end = Line break
    • Double returns = Paragraph
    • *Single asterisks* or _single underscores_ = Emphasis
    • **Double** or __double__ = Strong
    • This is [a link](http://the.link.example.com "The optional title text")
    For complete details on the Markdown syntax, see the Markdown documentation and Markdown Extra documentation for tables, footnotes, and more.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Plain text

  • HTML 태그를 사용할 수 없습니다.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 줄과 단락은 자동으로 분리됩니다.
댓글 첨부 파일
이 댓글에 이미지나 파일을 업로드 합니다.
파일 크기는 8 MB보다 작아야 합니다.
허용할 파일 형식: txt pdf doc xls gif jpg jpeg mp3 png rar zip.
CAPTCHA
이것은 자동으로 스팸을 올리는 것을 막기 위해서 제공됩니다.