表达式求值
给定一个表达式,其中运算符仅包含 +,-,,/(加 减 乘 整除),可能包含括号,请你求出表达式的最终值。
注意:
数据保证给定的表达式合法。
题目保证符号 - 只作为减号出现,不会作为负号出现,例如,-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;
}