栈和队列5:逆波兰表达式求值(c++代码实现)

什么是逆波兰表达式?

波兰表达式就是一种后缀表达式
后缀就是运算符卸载后面。
我们生活中常见的是中缀表达式:
比如: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
494
2942
/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;
    }
};
  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

辛伯达岛

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值