nyoj 1272 && 河南省第九届ACM竞赛A题

表达式求值

时间限制: 1000 ms  |  内存限制: 65535 KB
难度: 3
描述
假设表达式定义为: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
这个题与 nyoj 35 极度的相似,不同之处在于这个题中多了个Smax()表达式,然而这种表达式在 nyoj 305 中出现过,这就好说了......  两题的结合,诞生了这次省赛的这个题,原理大多数人都知道,但事实上这个题最后做出来的比较少,细节太多,很容易出错。。。

解题思路:方法同nyoj 35 ,思路基本完全一样,有一点不一样就是在处理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
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值