【第九届河南省大学生程序设计大赛】A题:表达式求值

问题描述:

假设表达式定义为:

1. 一个十进制的正整数 X 是一个表达式。

2. 如果 X 和 Y 是 表达式,则 X+Y, X*Y 也是表达式; *优先级高于+.

3. 如果 X 和 Y 是 表达式,则 函数 Smax(X,Y)也是表达式,其值为:先分别求出 X ,Y值的各位数字之和,再从中选最大数。

4.如果 X 是 表达式,则 (X)也是表达式。例如:表达式 12*(2+3)+Smax(333,220+280) 的值为 69。请你编程,对给定的表达式,输出其值。  

输入:

【标准输入】 第一行: T 表示要计算的表达式个数 (1≤ T ≤ 10) 接下来有 T 行, 每行是一个字符串,表示待求的表达式,长度<=1000

输出:

【标准输出】 对于每个表达式,输出一行,表示对应表达式的值。

样例输入:

3
12+2*3
12*(2+3)
12*(2+3)+Smax(333,220+280)

样例输出:

18
60
69

思路:

此题主要利用的是数据结构中栈的压入与弹出,把运算符和数据分别压入不同的栈中进行相应的操作,在处理Smax()时,遍历到 S 要把 S 压入到字符栈中,但是在压入之前,先压入一个左括弧 "(",目的是为了当遇到栈顶是 S 时,计算数据栈内的数据,即Smax() 表达式中的值。

题解代码:

#include<stdio.h>
#include<string.h>
#include<stack>
using namespace std;
stack<int> dsta;//数据栈 
stack<char> osta;//字符栈 
char s[1005];
int main()
{
	int T,i,j;
	scanf("%d",&T);
	while(T--)
	{
		scanf("\n%s",&s[1]);
		int len=strlen(&s[1]);
		s[0]='('; s[++len]=')';//在给定的表达式两端添加上一对括号 
		for(i=0;i<=len;i++)
		{
			if(s[i]=='(')
				osta.push(s[i]);
			else if(s[i]=='S')
			{
				osta.push('(');//压入一个左括弧 
				i+=3;
			}
			else if(s[i]>='0' && s[i]<='9')
			{
				int v=0;
				while(s[i]>='0' && s[i]<='9')
					v=v*10+(s[i++]-'0');
				i--;
				dsta.push(v);
			}
			else if(s[i]=='+')
			{
				while(osta.top()!='(' && osta.top()!=',')
				{
					int a=dsta.top(); dsta.pop();
					int b=dsta.top(); dsta.pop();
					int c;
					switch(osta.top())
					{
						case '+':c=b+a;break;
						case '*':c=b*a;break;
					}
					dsta.push(c);
					osta.pop();
				}
				osta.push(s[i]);
			}
			else if(s[i]=='*')
			{
				if(osta.top()=='*')
				{
					int a=dsta.top(); dsta.pop();
					int b=dsta.top(); dsta.pop();
					dsta.push(b*a);
					osta.pop();
				}
				osta.push(s[i]);
			}
			else if(s[i]==')' || s[i]==',')//遇到逗号及时求值 
			{ //当遇到 ','时,就把Smax的前半部分表达式的值求出来  
				while(osta.top()!='(')
				{
					int a=dsta.top(); dsta.pop();
					int b=dsta.top(); dsta.pop();
					int c,suma=0,sumb=0;
					switch(osta.top())
					{
						case '+':c=b+a;break;
						case '*':c=b*a;break;
						case ',':{
						//逐位分割求和 
							while(b!=0)
							{
								sumb+=b%10; b/=10;
							}
							while(a!=0)
							{
								suma+=a%10; a/=10;
							}
							c=max(suma,sumb);
							break;
						}
					}
					dsta.push(c);
					osta.pop();
				}
				osta.pop();
				if(s[i]==',')//求完值之后,把逗号压入栈内,以便后半部分Smax表达式的求值 
					osta.push(s[i]);
			}
		}
		printf("%d\n",dsta.top());
		dsta.pop();
	}
	return 0;
}

 

 

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值