洛谷 P1054 等价表达式

来源:互联网 发布:克里诺林裙淘宝 编辑:程序博客网 时间:2024/06/08 06:38

题目描述
明明进了中学之后,学到了代数表达式。有一天,他碰到一个很麻烦的选择题。这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也是一个代数表达式,题目的要求是判断选项中哪些代数表达式是和题干中的表达式等价的。
这个题目手算很麻烦,因为明明对计算机编程很感兴趣,所以他想是不是可以用计算机来解决这个问题。假设你是明明,能完成这个任务吗?
这个选择题中的每个表达式都满足下面的性质:
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年提高组第四题

#include<iostream>#include<cstring>#include<cstdio>#include<stack>#include<cmath>#include<algorithm>using namespace std;#define LL long long #define Mod 200600801char ch[60][500];int map[500];struct stack{    LL num[150101];    int topp;    inline bool empty(){ return topp==0 ? 1 : 0; }    inline void push(LL x){ num[++topp]=x; }    inline void pop(){ topp--; }    inline LL top(){ if(topp>=1) return num[topp]; }}st1,st2;LL pow(LL a,LL b){    LL ans=1,p=a;    for(;b;b>>=1,p*=p,p%=Mod)        if(b&1) ans*=p,ans%=Mod;    return ans;}inline void solve(){    int q=st2.top(); st2.pop();    if(q==1||q==2) return ;    LL num1=st1.top(); st1.pop();    LL num2=st1.top(); st1.pop();    if(q == 4) st1.push((((num2-num1)%Mod)+Mod)%Mod);    else if(q == 3) st1.push((num2+num1)%Mod);    else if(q == 5) st1.push((num2*num1)%Mod);    else if(q == 6) st1.push((pow(num2,num1))%Mod);    return;  }LL calc(char *ch1,int value){//计算某个表达式的值     while(!st1.empty()) st1.pop();    while(!st2.empty()) st2.pop();    LL num=0; bool ok=0;    for(int i=0;1;++i){        const char c=ch1[i];        if(c==' ') continue;        if(c=='a'){            st1.push(value);continue;        }        if(c>='0'&&c<='9') num=num*10+c-'0',ok=1;        else if(ok) num%=Mod,st1.push(num),num=0,ok=0;        if(c=='\0') break;        if(map[c]!=0){            int p=map[c];            if(p==1) st2.push(p);            else if(p==2){                while(st2.top()!=1){                    if(st2.top()==2) st2.pop();                    solve();                }                st2.pop();            } else {                while(!st2.empty()&&p<=st2.top()) solve();                st2.push(p);            }        }    }    while(!st2.empty()) solve();    return st1.top();}int main(){    map['(']=1;map[')']=2;map['+']=3;    map['-']=4;map['*']=5;map['^']=6;    gets(ch[0]);    int n;    scanf("%d",&n);    getchar(); getchar();    for(int i=1;i<=n;++i) gets(ch[i]);    for(int i=0;i<=n;++i){        int a1=0;        for(int j=0;ch[i][j]!='\0';++j){            if(ch[i][j]=='(') a1++;            if(ch[i][j]==')') {                if(a1>=1) a1--;                else ch[i][j]=' ';            }        }    }    LL num1=calc(ch[0],71);    LL num2=calc(ch[0],31);    for(int i=1;i<=n;++i){        LL num3=calc(ch[i],71),num4=calc(ch[i],31);        if(num1==num3&&num2==num4)            printf("%c",'A'+i-1);    }    return 0;}
原创粉丝点击