150.逆波兰表达式求值

记录新手小白的做题历程。


题目:

根据 逆波兰表示法,求表达式的值。

有效的算符包括 +、-、*、/ 。每个运算对象可以是整数,也可以是另一个逆波兰表达式。

注意 两个整数之间的除法只保留整数部分。

可以保证给定的逆波兰表达式总是有效的。换句话说,表达式总会得出有效数值且不存在除数为 0 的情况。

示例 1:

输入:tokens = ["2","1","+","3","*"]
输出:9
解释:该算式转化为常见的中缀算术表达式为:((2 + 1) * 3) = 9


思路:看到题目我是很迷惑的,啥是逆波兰表达式求值?

看了别人的博客,大致了解了一下。

总结:

1.逆波兰表达式就是后缀表达式,我们平时使用的是中缀表达式,运算符号在数字的中间

而题目给出了后缀表达式,我要通过这个表达式将答案算出来。

感觉不难,代码入下:

public:
    int evalRPN(vector<string>& tokens) {
        stack<string> stl;
        for(string c:tokens){
            char ch=c[0];
            if(!stl.empty()&&ch<'0'){//找到运算符,取栈顶两个元素进行计算
                int a=stl.top()-'0';
                stl.pop();
                int b=stl.top()-'0';
                stl.pop();
                if(ch=='+'){//进行符号判断运算
                    stl.push(a+b);
                }
                else if(ch=='-'){
                    stl.push(b-a);
                }
                else if(ch=='*'){
                    stl.push(a*b);
                }
                else{
                    stl.push(b/a);
                }
            }
            else{//如果不是运算符就放进栈中
                stl.push(ch);
            }
        }
        return stl.top()-'0';
    }
};

但是编译错误了 

 

我是不理解。 


官方解答:

逆波兰表达式严格遵循「从左到右」的运算。计算逆波兰表达式的值时,使用一个栈存储操作数,从左到右遍历逆波兰表达式,进行如下操作:

如果遇到操作数,则将操作数入栈;

如果遇到运算符,则将两个操作数出栈,其中先出栈的是右操作数,后出栈的是左操作数,使用运算符对两个操作数进行运算,将运算得到的新操作数入栈。

整个逆波兰表达式遍历完毕之后,栈内只有一个元素,该元素即为逆波兰表达式的值。

 思路是大体相同的,看看代码有哪些区别:

class Solution {
public:
    int evalRPN(vector<string>& tokens) {
        stack<int> stk;
        int n = tokens.size();
        for (int i = 0; i < n; i++) {
            string& token = tokens[i];
            if (isNumber(token)) {
                stk.push(atoi(token.c_str()));
            } else {
                int num2 = stk.top();
                stk.pop();
                int num1 = stk.top();
                stk.pop();
                switch (token[0]) {
                    case '+':
                        stk.push(num1 + num2);
                        break;
                    case '-':
                        stk.push(num1 - num2);
                        break;
                    case '*':
                        stk.push(num1 * num2);
                        break;
                    case '/':
                        stk.push(num1 / num2);
                        break;
                }
            }
        }
        return stk.top();
    }

    bool isNumber(string& token) {
        return !(token == "+" || token == "-" || token == "*" || token == "/");
    }
};

大概是我没有处理好他传过来的vector<string>& tokens


其他人的处理方式:

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 {
                st.push(stoi(tokens[i]));
            }
        }
        int result = st.top();
        st.pop(); // 把栈里最后一个元素弹出(其实不弹出也没事)
        return result;
    }
};

他直接用字符串去判断的,没有转化成字符,而计算的时候就直接计算。 

而且还在这里发现了一个自己不懂的东西。

stoi,这是一个将n进制的字符串转化为数字的函数。本来若是字符,可以之间减去‘0’,而这里是字符串,就采用了这个函数。
stoi链接


还算比较简单吧,就是数据要处理好。

下一题!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值