什么是逆波兰表达式?
波兰表达式就是一种后缀表达式
后缀就是运算符卸载后面。
我们生活中常见的是中缀表达式:
比如:1+2
这里 + 写在1和2之间。
后缀表达式呢,会这样写:
1 2 +
为什么用后缀表达式(也就是逆波兰表达式?)
原因很简单,逆波兰表达式是给计算机看的,逆波兰表达式更方便计算机理解。
中缀表达式1+2这种的,更方便人理解,但是计算机不好理解。
为啥计算机好理解逆波兰表达式,不好理解中缀表达式呢?
首先我们看看逆波兰表达式是怎么工作的。
比如现在有这么一个中缀式
(1+2)*3-4/2转换为逆波兰式子
( ( 1 2 + ) 3 * ) 4 2 / -
怎么计算的呢?
把这个式子存放在字符串中。
然后,从左往右依次扫描这个字符串。
如果,遇到数字,则让它进入栈中。
如果,遇到运算符,则把栈顶的两个数字拉出来,用那个运算符做运算,然后再把结果push到栈中。
以上面这个式子为例,模拟下运算流程:
扫描到哪个字符 | 栈中的元素 |
---|---|
( | ( |
( | (( |
1 | ((1 |
2 | ((12 |
+ | ((3 |
) | (3 |
3 | (33 |
* | (9 |
) | 9 |
4 | 94 |
2 | 942 |
/ | 92 |
- | 7 |
好啦,最后结果就是7。
注意:
1.如果两个括号遇见会消去
2.这里计算机不用去判断1和2应该去做什么运算,因为,使用逆波兰表达式的时候,1和2后面就是+,只能进行加法运算。所以,其实有没有括号根本没有任何影响。
完全不用考虑优先级这个问题,就是无脑算!!!
但是,如果是中缀表达式,1+2遇见加号,算还是不算?还要判断优先级,很麻烦。
所以,一般逆波兰表达式都写成下面这样
1 2 + 3 * 4 2 -
没有括号,不用考虑优先级,直接从左往右扫描就完事了。
下面是cpp代码实现
#include <stack>
#include <string>
#include <iostream>
using namespace std;
class Solution {
public:
int evalRPN(vector<string>& tokens) {
stack<int> st;//建立一个整数栈
for (int i = 0; i < tokens.size(); i++) {
if (tokens[i] == "+" || tokens[i] == "-" || tokens[i] == "*" || tokens[i] == "/") {
//若此时扫描到字符串中的内容是运算符号
//那么就弹出栈最顶端的两个元素,对其进行该操作符的运算
int num1 = st.top();
st.pop();
int num2 = st.top();
st.pop();
//注意,除法和减法要注意运算顺序,谁在前,谁在后
if (tokens[i] == "+") st.push(num2 + num1);
if (tokens[i] == "-") st.push(num2 - num1);
if (tokens[i] == "*") st.push(num2 * num1);
if (tokens[i] == "/") st.push(num2 / num1);
} else {
//如果扫描到的不是运算符,而是数字,那就直接压入栈中
//stoi()函数:把string类型转化为int类型
st.push(stoi(tokens[i]));
}
}
int result = st.top();
st.pop();
return result;
}
};