表达式求值

表达式求值

给定一个表达式,其中运算符仅包含 +,-,,/(加 减 乘 整除),可能包含括号,请你求出表达式的最终值。
注意:
数据保证给定的表达式合法。
题目保证符号 - 只作为减号出现,不会作为负号出现,例如,-1+2,(2+2)
(-(1+1)+2) 之类表达式均不会出现。
题目保证表达式中所有数字均为正整数。
题目保证表达式在中间计算过程以及结果中,均不超过 231−1。
题目中的整除是指向 0 取整,也就是说对于大于 0 的结果向下取整,例如 5/3=1,对于小于 0 的结果向上取整,例如 5/(1−4)=−1。
C++和Java中的整除默认是向零取整;Python中的整除//默认向下取整,因此Python的eval()函数中的整除也是向下取整,在本题中不能直接使用。
输入格式
共一行,为给定表达式。

输出格式
共一行,为表达式的结果。

数据范围
表达式的长度不超过 105。

输入样例:
(2+2)*(1+1)
输出样例:
8
思路分析:
表达式求值关键在几个点
1.是开两个栈,一个是操作数栈,一个是运算符栈
2。即将进栈的运算符和运算符栈的栈顶运算符进行优先级比较,如果刚要进栈的运算符优先级小于栈顶,则先出栈计算,然后在将运算符入栈。如果刚要进栈的运算符优先级高,则直接进栈
3.我们知道加法和减法的运算符优先级一样,乘法和除法的运算符优先级一样,同时乘除法的优先级大于加减法,所以我们规定乘除法优先级为2,加减法优先级为1,同时我们用一个哈希表来存储
4.在表达式中如果有括号要先计算括号,所以当刚要进栈的运算符是’(’时,直接进运算符栈,如果是’)’,则这时需要弹出元素计算,找到遇到‘(’,然后将‘(’弹出,将运算结果入操作数栈
时间复杂度为O(n),因为我们只需要从左往右搜描一遍表达式即可

#include<iostream>
#include<stack>
#include<string>
#include<unordered_map>
#include<algorithm>
using namespace std;
stack<int>num;
stack<char>op;
unordered_map<char,int>pr{{'+',1},{'-',1},{'*',2},{'/',2}};//运算符优先级哈希表
void eval(){//运算函数
    auto b=num.top();num.pop();
    auto a=num.top();num.pop();//依次弹出两个操作数
    auto c=op.top();op.pop();//弹出一个运算符
    int x=0;
    if(c=='+')x=a+b;
    else if(c=='-')x=a-b;
    else if(c=='*')x=a*b;
    else
    x=a/b;
    num.push(x);//运算结果入操作数栈
}
int main(){
    string str;
    cin>>str;
    for(int i=0;i<str.size();i++){
        auto c=str[i];
        if(isdigit(c)){//判断当前字符是否是数字的库函数
            int x=0,j=i;
            while(isdigit(str[j])&&j<str.size())//一个数可能是多位数,所以要得到这个数
            x=x*10+str[j++]-'0';
            i=j-1;//移动i的位置
            num.push(x);//将该数入操作数栈
        }
        else if(c=='(')//为'('直接入运算符栈
        op.push(c);
        else if(c==')'){
            while(op.top()!='(')eval();
            op.pop();
        }
        else{
            while(op.size()&&pr[op.top()]>=pr[c]&&op.top()!='(')eval();//栈顶运算符优先级高并且栈不空进行运算
            op.push(c);//运算完后当前字符入运算符栈
        }
    }
    while(op.size())//转换完后进行计算,找到运算符栈为空
    eval();
    cout<<num.top()<<endl;//最后结果在操作数栈中
    
    return 0;
}
  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值