逆波兰表示法

逆波兰表示法(Reverse Polish notation,RPN,或逆波兰记法),是一种是由波兰数学家扬·武卡谢维奇1920年引入的数学表达式方式,在逆波兰记法中,所有操作符置于操作数的后面,因此也被称为后缀表示法。

比如下列中缀表达式:

(1-2)*(4+5)

采用逆波兰表示法表示为:

12-45+*

逆波兰表示法中不需要圆括号,只要知道每个运算符需要几个操作数就不会引起歧义。

 

中缀表达式转后缀表达式规则

中缀表达式a + b*c + (d * e + f) * g,其转换成后缀表达式则为a b c * + d e * f  + g * +。

转换过程需要用到栈,具体过程如下:

1)如果遇到操作数,我们就直接将其输出。

2)如果遇到操作符,则我们将其放入到栈中,遇到左括号时我们也将其放入栈中。

3)如果遇到一个右括号,则将栈元素弹出,将弹出的操作符输出直到遇到左括号为止。注意,左括号只弹出并不输出。

4)如果遇到任何其他的操作符,如(“+”, “*”,“(”)等,从栈中弹出元素直到遇到发现更低优先级的元素(或者栈为空)为止。弹出完这些元素后,才将遇到的操作符压入到栈中。有一点需要注意,只有在遇到" ) "的情况下我们才弹出" ( ",其他情况我们都不会弹出" ( "。也就是说这种操作," + "的优先级最低," ( "优先级最高。

5)如果我们读到了输入的末尾,则将栈中所有元素依次弹出。

 

Program Source

 

#include <iostream>
#include <string.h>
#include <stack>
#include <stdlib.h>

using namespace std;

char symbol[] = "+-*/(";

int getSymbol(char c)
{
        int i;
        for(i=0; i<strlen(symbol); i++)
        {
                if(symbol[i]== c)
                        break;
        }
        return i;
}

int getInt(char *p)
{
        int i=0;
        while(isdigit(*p))
        {
                i = i*10+(*p-'0');
                p++;
        }
        return i;
}

int main(int argc, char* argv[])
{
        char exp[]="1+2*3+(4*5+6)*7";
        char *p=exp;

        int t;
        stack<char> op;

        while(true)
        {
                if(isdigit(*p))
                {
                        t = getInt(p);
                        cout<<t<<endl;
                }

                if(*p==')')
                {
                        while(op.top()!='(')
                        {
                                cout<<op.top()<<endl;
                                op.pop();
                        }
                        op.pop();
                }
                else if(*p=='(')
                {
                        op.push('(');
                }
                else if(*p=='+'||*p=='-'||*p=='*'||*p=='/')
                {
                        while(!op.empty()&&getSymbol(*p)<=getSymbol(op.top()))
                        {
                                if(op.top()!='(')
                                {
                                        cout<<op.top()<<endl;
                                        op.pop();
                                }
                                else
                                        break;
                        }
                        op.push(*p);
                }
                else if(*p=='\0')
                        break;

                p++;
        }

        while(!op.empty())
        {
                cout<<op.top()<<endl;
                op.pop();
        }

        return 0;
}

 

 

前面提过的中缀表达式a + b*c + (d * e + f) * g,其转换成后缀表达式则为a b c * + d e * f  + g * +

 

程序中的中缀表达式为1+2*3+(4*5+6)*7,其转换成后缀表达式则为123*+45*6+7*+

验证结果

 

[yangtze@localhost ~]$ g++ -o postfixExp postfixExp.cpp 
[yangtze@localhost ~]$ ./postfixExp 
1
2
3
*
+
4
5
*
6
+
7
*
+
[yangtze@localhost ~]$ 

 

 

 

posted @ 2013-10-29 22:36  艾丽娅的猫  阅读(1068)  评论(0编辑  收藏  举报