P1054 等价表达式

题目描述

明明进了中学之后,学到了代数表达式。有一天,他碰到一个很麻烦的选择题。这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也是一个代数表达式,题目的要求是判断选项中哪些代数表达式是和题干中的表达式等价的。

这个题目手算很麻烦,因为明明对计算机编程很感兴趣,所以他想是不是可以用计算机来解决这个问题。假设你是明明,能完成这个任务吗?

这个选择题中的每个表达式都满足下面的性质:

1. 表达式只可能包含一个变量‘a’。

2. 表达式中出现的数都是正整数,而且都小于10000。

3. 表达式中可以包括四种运算‘+’(加),‘-’(减),‘*’(乘),‘^’(乘幂),以及小括号‘(’,‘)’。小括号的优先级最高,其次是‘^’,然后是‘*’,最后是‘+’和‘-’。‘+’和‘-’的优先级是相同的。相同优先级的运算从左到右进行。(注意:运算符‘+’,‘-’,‘*’,‘^’以及小括号‘(’,‘)’都是英文字符)

4. 幂指数只可能是1到10之间的正整数(包括1和10)。

5. 表达式内部,头部或者尾部都可能有一些多余的空格。

下面是一些合理的表达式的例子:

((a^1) ^ 2)^3,a*a+a-a,((a+a)),9999+(a-a)*a,1 + (a -1)^3,1^10^9……

输入输出格式

输入格式:

 

输入文件equal.in的第一行给出的是题干中的表达式。第二行是一个整数n(2 <= n <= 26),表示选项的个数。后面n行,每行包括一个选项中的表达式。这n个选项的标号分别是A,B,C,D……

输入中的表达式的长度都不超过50个字符,而且保证选项中总有表达式和题干中的表达式是等价的。

 

输出格式:

 

输出文件equal.out包括一行,这一行包括一系列选项的标号,表示哪些选项是和题干中的表达式等价的。选项的标号按照字母顺序排列,而且之间没有空格。

 

输入输出样例

输入样例#1:
( a + 1) ^2
3
(a-1)^2+4*a
a + 1+ a
a^2 + 2 * a * 1 + 1^2 + 10 -10 +a -a
输出样例#1:
AC

说明

对于30%的数据,表达式中只可能出现两种运算符‘+’和‘-’;

对于其它的数据,四种运算符‘+’,‘-’,‘*’,‘^’在表达式中都可能出现。

对于全部的数据,表达式中都可能出现小括号‘(’和‘)’。

2005年提高组第四题

洛谷P1054 等价表达式_优先级洛谷P1054 等价表达式_i++_02
#include<iostream>
#include<cmath>
#include<cstring>
#include<cstdio>
using namespace std;
int n,a[7]={2,1,4,5,9,0,7};
int ans[7];
char sst[101];
int Pow(int a,int b){
    int res=1;
    while(b){
        if(b&1)res=res*a;
        a=a*a;
        b>>=1;
    }
    return res;
}
int number[101],i=0,p=1;
char symbol[101],s[256],t[256];
void push(){symbol[++p]=s[i];}
void pop(){
    switch(symbol[p--]){
        case'+':number[p]+=number[p+1];break;
        case'-':number[p]-=number[p+1];break;
        case'*':number[p]*=number[p+1];break;
        case'/':number[p]/=number[p+1];break;
        case'^':number[p]=Pow(number[p],number[p+1]);break;
    }
}
bool can(){
    if((s[i]=='+'||s[i]=='-')&&symbol[p]!='(')return 1;
    if((s[i]=='*'||s[i]=='/')&&(symbol[p]=='*'||symbol[p]=='/'))return 1;
    if(s[i]=='^'&&symbol[p]=='^')return 1;
    return 0;
}
int Calc(int len){
    //cout<<s<<endl;
    p=1;i=0;
    memset(number,0,sizeof(number));
    memset(symbol,0,sizeof(symbol));
    s[len-1]=')';symbol[p]='(';
    while(i<len){
        while(s[i]==' ')i++;
        while(s[i]=='('){
            push();i++;
            while(s[i]==' ')i++;
        }
        int x=0;
        while(s[i]>='0'&&s[i]<='9'){
            x=x*10+s[i]-'0';
            i++;
            while(s[i]==' ')i++;
        }
        number[p]=x;
        int j=0;
        do{
            if(s[i]==')'){
                while(symbol[p]!='(')pop();
                number[--p]=number[p+1];
            }
            else{
                while(can())pop();
                push();
            }
            j=0;
            i++;j++;
            while(s[i]==' ')i++,j++;
        }while(i<strlen(s)&&s[i-j]==')');
    }
    return number[0];
}

int main(){
    freopen("Cola.txt","r",stdin);
    gets(sst);
    for(int j=0;j<7;j++){
        for(int k=0;k<strlen(sst);k++){
            if(sst[k]=='a')s[k]=(char)(a[j]+'0');
            else s[k]=sst[k];
        }
        ans[j]=Calc(strlen(sst)+1);
        //printf("%d ",ans[j]);
    }
    scanf("%d",&n);
    gets(sst);
    for(int t=0;t<n;t++){
        gets(sst);
        bool flag=0;
        for(int j=0;j<7;j++){
            for(int k=0;k<strlen(sst);k++){
                if(sst[k]=='a')s[k]=(char)(a[j]+'0');
                else s[k]=sst[k];
            }
            if(Calc(strlen(sst)+1)!=ans[j]){flag=1;break;}
        }
        if(!flag){
            printf("%c",t+'A');
        }
    }
    return 0;
}
40分 TLE