带括号的四则运算表达式的求值(栈实现)
利用栈这种数据结构来实现一个加减乘除以及带括弧的混合数学表达式的计算,对于数学表达式的计算,可以设置一个运算符栈和一个数字栈,分别来保存运算符、数字或者中间计算得到的结果。将整个表达式看做一个字符串,从开头依次判断每个字符是运算符还是数字,若是运算符,则根据运算符优先级来确定是将其压栈还是弹栈进行计算;若是数字,则先将其转化并计入一个临时double型变量中,看下一个字符是否为运算符栈,若是,则将临时变量压进数字栈,否则读取下一个数字字符并进行相关处理后累加到临时变量中,直到下一个字符为运算符,将临时变量压进数字栈。最后,当字符为"="时,结束计算,得到计算结果。本算法需要先设置一个运算符优先级表,下表可供参考:
+ | - | * | / | ( | ) | = | |
+ | > | > | < | < | < | > | > |
- | > | > | < | < | < | > | > |
* | > | > | > | > | < | > | > |
/ | > | > | > | > | < | > | > |
( | < | < | < | < | < | = | |
) | > | > | > | > | > | > | |
= | < | < | < | < | < | = |
参考代码如下:
注:本代码输入表达式时末尾不需要“=”,程序中会自动添加
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <stack>
#include <cstring>
using namespace std;
char Precede(char a, char b) { //判断运算符优先级
int i, j;
char Table[8][8] = {
{' ','+','-','*','/','(',')','='},
{'+','>','>','<','<','<','>','>'},
{'-','>','>','<','<','<','>','>'},
{'*','>','>','>','>','<','>','>'},
{'/','>','>','>','>','<','>','>'},
{'(','<','<','<','<','<','=',' '},
{')','>','>','>','>',' ','>','>'},
{'=','<','<','<','<','<',' ','='}
}; //优先级表格
for(i=0; i<8; i++)
if(Table[0][i]==a) //寻找运算符a
break;
for(j=0; j<8; j++) //寻找运算符b
if(Table[j][0]==b)
break;
return Table[j][i];
}
bool Calcu_temp(double a, char theta, double b, double &r) { //计算二元表达式的值
if(theta=='+')
r = a + b;
else if(theta=='-')
r = a - b;
else if(theta=='*')
r = a * b;
else {
if(fabs(b-0.0)<1e-8) //如果除数为0,返回错误信息
return false;
else
r = a / b;
}
return true;
}
bool IsOper(char ch) { //判断字符ch是否为运算符
char ptr[10] = {'+', '-', '*', '/', '(', ')', '='};
int i;
for(i=0; i<7; i++) {
if(ch==ptr[i])
return true;
}
return false;
}
bool Calculate(char s[], double &result) { //计算表达式的结果
char theta;
int i = 0, j, point = 0;
double a, b, r, num = 0;
stack<double> num_stack; //数字栈
stack<char> oper_stack; //运算符栈
oper_stack.push('=');
while(s[i]!='=' || oper_stack.top()!='=') { //对表达式a进行计算
if((s[i]>='0' && s[i]<='9') || s[i]=='.') { //字符是数字或者小数点
num = 0; //初始化数字为0
point = 0; //point用来标记是否出现小数点以及当前处于小数点后第x位,point==10^x
if(s[i]=='.')
point = 10;
else
num = s[i] - 48;
j = i + 1;
while(!IsOper(s[j])) { //继续往后查找并记录该数字,直到该数字结束遇到运算符为止
if(s[j]=='.') {
point = 10;
j++;
continue;
}
if(!point) //整数部分
num = num * 10 + ( s[j] - 48 );
else {
num = num + 1.0 * ( s[j] - 48 ) / point; //小数部分
point *= 10; //小数位数后移一位
}
j++;
}
i = j;
num_stack.push(num); //将该数字压入栈中
}
else if(IsOper(s[i])) { //字符是运算符
switch(Precede(s[i],oper_stack.top())) { //该运算符和栈顶运算符进行优先级比较并做相关处理
case '<':
oper_stack.push(s[i++]);
break;
case '=':
oper_stack.pop();
i++;
break;
case '>':
theta = oper_stack.top(); //从栈中弹出一个运算符进行计算
oper_stack.pop();
b = num_stack.top(); //弹出两个数字,注意顺序,先弹出的数是第二个操作数
num_stack.pop();
a = num_stack.top();
num_stack.pop();
if ( Calcu_temp(a, theta, b, r) ) //计算并判断是否有除数等于0的情况
num_stack.push(r); //若正常,则将结果压入栈中
else
return false; //出现除数为0的情况,返回错误信息
break;
}
}
}
result = num_stack.top(); //最后数字栈中的数即为表达式的最终结果
return true;
}
bool Check(char s[]) { //检查表达式括号是否匹配
int flag=0, i;
for(i=0; s[i]!=0; i++) {
if(s[i]=='(')
flag++;
if(s[i]==')')
flag--;
}
if(flag)
return false;
else
return true;
}
int main() {
// freopen("in.txt", "r", stdin);
int i, j;
char s1[210], s2[210];
double result;
while(gets(s1)!=NULL) { //输入表达式
if(strlen(s1)==1 && s1[0]=='0')
break;
for(i=0,j=0; s1[i]!=0; i++) { //将表达式转换为规格化的表达式,并在末尾加上“=”,保存在s2中
if(s1[i]==' ')
continue;
s2[j++] = s1[i];
}
s2[j++] = '=';
s2[j] = '\0';
if(Check(s2)) { //检查括号是否匹配
if(Calculate(s2, result)) //计算并检查表达式中是否出现除数为0的情况
printf("%lf\n",result);
else
printf("The divisor cannot be zero!\n");
}
else
printf("The parentheses do not match!\n");
}
return 0;
}
/*
in.txt:
1 + 2 / ( 3 - 2.2 / 1.10 ) + 5
4 + 2.03 * .05 - 7 / 3.5
2 + 3. - .1
1 + 2 / ( 2 - 2.2 / 1.1 ) - 3
2 * ( ( 3 + 2 ) - 3
0
*/
本代码生成的程序,在输入表达式时,数字与运算符之间可以有若干空格或者没有;对于小数,可以没有整数部分(即小数点之前没有数字),也可以没有小数部分(即小数点之后没有数字);表达式可以为无效表达式(如括号不匹配或者出现除数为0的情况等);如以上的样例输入。对于以上样例输入,其对应输出结果为:
8.000000
2.101500
4.900000
The divisor cannot be zero!
The parentheses do not match!