-
题目描述:
-
读入一个只包含 +, -, *, / 的非负整数计算表达式,计算该表达式的值。
-
输入:
-
测试输入包含若干测试用例,每个测试用例占一行,每行不超过200个字符,整数和运算符之间用一个空格分隔。没有非法表达式。当一行中只有0时输入结束,相应的结果不要输出。
-
输出:
-
对每个测试用例输出1行,即该表达式的值,精确到小数点后2位。
-
样例输入:
-
1 + 2 4 + 2 * 5 - 7 / 11 0
-
样例输出:
-
3.00 13.36
-
来源:
注:表达式的运算是栈的一个经典应用,利用这个题来熟悉栈
代码如下:
#include <stdio.h>
#include <stack>
using namespace std;
stack<int> op;
stack<double> in;
char str[201];
int mat[5][5]={
/* + - * / */
/*0标记*/ 1,0,0,0,0,
/* + */ 1,0,0,0,0,
/* - */ 1,0,0,0,0,
/* * */ 1,1,1,0,0,
/* / */ 1,1,1,0,0,
};
void GetOp(bool &reto,int &retn,int &i)
{ //若i=0且运算符栈为空,则添加一个人为的起止标记符 0,便于最后判断是否结束
if(i==0&&op.empty()==true)
{
reto=true;
retn=0;
return;
}
//如果str[i]==0,即为Null,说明字符扫描到头,这时要再添加一个起止标记符 0 ,这样堆栈中有两个 0,满足条件跳出循环
if(str[i]==0)
{
reto=true;
retn=0;
return;
}
//如果为数字 ,先标记为false
if(str[i]>'0'&&str[i]<'9')
{
reto=false;
}
else
{
reto=true;
switch(str[i])
{
case '+':
retn=1;
break;
case '-':
retn=2;
break;
case '*':
retn=3;
break;
case '/':
retn=4;
break;
}
i+=2;//跳过本字符和下一个空格
return;
}
//继续数字的情况,因为数字可能为多位数,此时连续扫描单个字符,要计算数字的真值
retn=0;
for(;str[i]!=' '&&str[i]!=0;i++)
{
retn*=10;
retn+=str[i]-'0';
}
//如果str[i]为空格则还未遍历完,修改i的值以继续遍历
if(str[i]==' ') i++;
return;
}
int main(int argc, char** argv) {
while(gets(str))//输入字符串,当其位于文件尾时,gets返回0
{
if(str[0]=='0'&&str[1]==0) break;//当输入只有一个0时结束循环
bool IsOp;
int dat;
int i=0;
while(!op.empty()) op.pop();
while(!in.empty()) in.pop();
while(true)
{
GetOp(IsOp,dat,i);
//如果是数字就直接压栈
if(IsOp==false)
{
in.push(dat);
}
else
{ //如果遍历的为第一个字符(把起止标记符压栈)或者下个运算符优先级高于栈顶运算符 ,则直接入栈
if(op.empty()==true||mat[dat][op.top()]==1)
{
op.push(dat);
}
//如果优先级低于栈顶
else
{
while(mat[dat][op.top()]==0)
{
double a=in.top();
in.pop();
double b=in.top();
in.pop();
int cal = op.top();
op.pop();
if(cal==1) in.push(b+a);
else if(cal==2) in.push(b-a);
else if(cal==3) in.push(b*a);
else in.push(b/a);
//这里一定要注意a,b 的前后顺序,谁是先出栈的谁是被加/减/乘/除数
}
//把运算符压栈
op.push(dat);
}
}
//如果只剩下两个起止符号,说明已经结束
if(op.size()==2&&op.top()==0)
break;
}
printf("%.2lf\n",in.top());
}
return 0;
}