后缀表达式求值

题目要求:

后缀表达式求值:建立一个操作数栈S。然后从左到右读表达式,如果读到操作数就将它压入栈S中,如果读到n元运算符(即需要参数个数为n的运算符)则取出由栈顶向下的n项操作数进行运算,再将运算的结果代替原栈顶的n项压入栈中。重复上面过程,如果后缀表达式读完且栈中只剩一个操作数,则该数就是运算结果;如果后缀表达式读完但是栈中操作数多于一个,则后缀表达式错误;如果栈中操作数只剩一个,但是后缀表达式还未读完且当前运算符为双元操作符,则后缀表达式同样错误。
输入格式:
在一行中输入一个以#号结束的非空后缀式,#不属于表达式的一部分,操作数和运算符都以空格分隔,运算数为绝对值不超过100的整数,运算符仅有+、-、*、/ 四种。
输出格式:
输出后缀式计算结果,所有的计算都只取结果的整数部分。题目保证计算的中间和最后结果的绝对值都不超过109。如果执行除法时出现分母为零的非法操作,则在一行中输出:Error: X/0,X是当时的分子。如果后缀表达式中运算符多了或者少了,则在一行中输出:Expression Error: X,X是当时栈顶元素。
输入样例1:5 -2 + 3 * #  输出:9
输入样例2:5 -2 2 + / #   输出:Error: 5/0
输入样例3:5 -1 3 + / - * #   输出Expression Error: 2

框架结构:

//用于存放操作数的栈
int OPND[100];
int top=0;
//运算操作
int operate(int a,char operate,int b)
//计算表达式函数,如果出现错误的表达式返回false,表达式正确返回true,并且表达式的值最终会存在栈里。
bool caculate(string s);
int main()
{
	string s;
	top=0;
	getline(cin,s);//可以接受空格 
	if(caculate(s))
	{
		cout<<"表达式的值为:"<<OPND[0];                                                 
	}
}

栈的一些操作:

其实不用特意的写这些函数,下面几个操作都可以用一个语句完成,写成函数是为了方便阅读

//栈的操作
void push(int num)
{
	OPND[top++]=num;
}
int pop()
{
	return OPND[--top];
}
int GetTop()
{
	return OPND[top-1];
}

将两个数进行一次操作的函数:

int operate(int a,char operate,int b)
{//不出现/0的情况 
    int ans;
	if(operate=='+')
	    ans=a+b;
	else if(operate=='-')
	    ans=a-b;
	else if(operate=='*')
	    ans=a*b;
	else
	    ans=a/b;
	return ans;
}

计算表达式的函数:

我是一个字符一个字符扫描的,有的人习惯将表达式的string串以空格分成多个string串,对每个string串扫描,这样也可以。

bool caculate(string s)
{
	int a,b,num=0,i=0;
	int sign=1;//记录数字符号 
	char thea;
	while(s[i]!='#')
	{
		if(s[i]>='0'&&s[i]<='9')
		{//遇到数字开始构造 
			num=10*num+s[i]-'0';
		}
		if(s[i]=='-'&&s[i+1]>='0'&&s[i+1]<='9')
		{//'-'后面跟着数字,说明遇到了负数 ,标记符号 
			sign=-1;
		}
		else if(s[i]=='+'||s[i]=='-'||s[i]=='/'||s[i]=='*')
		{
			if(top==0)
			{//没有操作数 
				cout<<"Expression Error: No operand!";
				return false;
			}
			else if(top<2)
			{//操作数不够 
				cout<<"Expression Error: "<<OPND[top-1];
				return false; 
			}
			b=pop();
			a=pop();
			if(b==0&&s[i]=='/')
			{//除数为零的情况 
				cout<<"Expression Error: "<<a<<"/"<<b;
				return false;
			}
			int ans=operate(a,s[i],b);//先抛出的做第二操作数
			push(ans); 
		}
		else if(s[i-1]>='0'&&s[i-1]<='9'&&s[i]==' ')
		{//当前字符是空格并且前面字符是数字 
			num*=sign;
			push(num);
			num=0;
			sign=1;
		}
		i++;
	}
	if(top>1)
	{//扫描结束后栈里的数大于一个,说明表达式有误
		cout<<"Expression Error: "<<OPND[top-1];
		return false;
	}
	return true;
}

代码:

#include<iostream>
using namespace std;
int OPND[100];
int top=0;
//栈的操作
void push(int num)
{
	OPND[top++]=num;
}
int pop()
{
	return OPND[--top];
}
int GetTop()
{
	return OPND[top-1];
}
int operate(int a,char operate,int b)
{//不出现/0的情况 
    int ans;
	if(operate=='+')
	    ans=a+b;
	else if(operate=='-')
	    ans=a-b;
	else if(operate=='*')
	    ans=a*b;
	else
	    ans=a/b;
	return ans;
}
bool caculate(string s)
{
	int a,b,num=0,i=0;
	int sign=1;//记录数字符号 
	char thea;
	while(s[i]!='#')
	{
		if(s[i]>='0'&&s[i]<='9')
		{//遇到数字开始构造 
			num=10*num+s[i]-'0';
		}
		if(s[i]=='-'&&s[i+1]>='0'&&s[i+1]<='9')
		{//'-'后面跟着数字,说明遇到了负数 ,标记符号 
			sign=-1;
		}
		else if(s[i]=='+'||s[i]=='-'||s[i]=='/'||s[i]=='*')
		{
			if(top==0)
			{//没有操作数 
				cout<<"Expression Error: No operand!";
				return false;
			}
			else if(top<2)
			{//操作数不够 
				cout<<"Expression Error: "<<OPND[top-1];
				return false; 
			}
			b=pop();
			a=pop();
			if(b==0&&s[i]=='/')
			{//除数为零的情况 
				cout<<"Expression Error: "<<a<<"/"<<b;
				return false;
			}
			int ans=operate(a,s[i],b);//先抛出的做第二操作数
			push(ans); 
		}
		else if(s[i-1]>='0'&&s[i-1]<='9'&&s[i]==' ')
		{//当前字符是空格并且前面字符是数字 
			num*=sign;
			push(num);
			num=0;
			sign=1;
		}
		i++;
	}
	if(top>1)
	{
		cout<<"Expression Error: "<<OPND[top-1];
		return false;
	}
	return true;
}
int main()
{
	string s;
	top=0;
	getline(cin,s);//可以接受空格 
//	cout<<s;
	if(caculate(s))
	{
		cout<<"表达式的值为:"<<OPND[0];                                                 
	}
}

  • 10
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值