明明进了中学之后,学到了代数表达式。有一天,他碰到一个很麻烦的选择题。这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也是一个代数表达式,题目的要求是判断选项中哪些代数表达式是和题干中的表达式等价的。这个题目手算很麻烦,因为明明对计算机编程很感兴趣,所以他想是不是可以用计算机来解决这个问题。假设你是明明,能完成这个任务吗?这个选择题中的每个表达式都满足下面的性质: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……对于30%的数据,表达式中只可能出现两种运算符‘+’和‘-’;对于其它的数据,四种运算符‘+’,‘-’,‘*’,‘^’在表达式中都可能出现。对于全部的数据,表达式中都可能出现小括号‘(’和‘)’。分析:


1.不需要考虑括号不匹配问题,输入绝对合法。
2.不需要考虑形如
    (-3+a^7)*2
  这样的情况。

3.另一方面,我认为只有试够11个数才能充分说明正确性:因为

a^10+k9a^9+k8a^8+.......=(a+3)^10+......

这是一个一元十次方程,它需要11个点来确定一条十次曲线。所以要将0-10都代入才能说明问题。

当然,如果这是一条6次曲线,7个点就够,可是谁有想写一个判断次数的函数呢?

4.因为取余运算,导致运算顺序不同,结果就不同。这是一个神坑。所以试的点少一点	,取余数大一点就容易过。



#include<iostream>
using namespace std;
#define big 10007
#define test 8
char ti[51];//题干
int size;//选项个数
int result[test];//准备试的数
bool prior(char a, char b){//运算符a优先级是否大于b
	if (a == '^')return true;
	if (b == '^')return false;
	if (a == '*')return true;
	if (b == '*')return false;
	return true;
}
int power(int a, int b){//乘幂函数要取余
	int i;
	int ans = 1;
	for (i = 0; i < b; i++){
		ans *= a;
		ans %= big;
	}
	return ans;
}
int op(int a, int b, char o){
	switch (o){
	case '^':return power(a, b);
	case '*':a *= b; a %= big; return a;
	case '+':return (a + b) % big;
	case '-':return (a - b) % big;
	}
}
//x表示数字,o表示符号
int calculate(int x[], int xsize, char o[], int osize){
	int i;
	int xstack[51];
	int xtop = 0;
	char ostack[51];
	int otop = 0;
	int xi, oi;
	xi = oi = 0;
	xstack[xtop++] = x[xi++];
	while (xi<xsize&&oi<osize){
		while (otop != 0 && prior(ostack[otop - 1], o[oi])){
			xstack[xtop - 2] = op(xstack[xtop - 2], xstack[xtop - 1], ostack[otop - 1]);
			xtop--;
			otop--;
		}
		ostack[otop++] = o[oi++];
		xstack[xtop++] = x[xi++];
	}
	while (otop >0){
		xstack[xtop - 2] = op(xstack[xtop - 2], xstack[xtop - 1], ostack[otop - 1]);
		xtop--;
		otop--;
	}
	return xstack[0];
}
int go(char ex[], int a){
	int i, j;
	int x[51];
	int xi = 0;
	char o[51];
	int oi = 0;
	i = 0;
	while (ex[i]){
		while (ex[i] == ' ')i++;
		if (ex[i] == 0)break;
		if (ex[i] == '('){
			int left = 1;
			char temp[51];
			j = 0;
			i++;
			while (true){
				temp[j] = ex[i];
				if (temp[j] == '(')left++;
				else if (temp[j] == ')')left--;
				if (left == 0){
					temp[j] = 0;
					x[xi++] = go(temp, a);
					i++;
					break;
				}
				i++; j++;
			}
		}
		else if (ex[i] == 'a'){
			x[xi++] = a;
			i++;
		}
		else{
			int n = 0;
			while (ex[i] >= '0'&&ex[i] <= '9'){
				n *= 10;
				n += ex[i] - '0';
				i++;
			}
			x[xi++] = n;
		}
		while (ex[i] == ' ')i++;
		if (ex[i] == 0)break;
		if (ex[i] == '+' || ex[i] == '-' || ex[i] == '*' || ex[i] == '^'){
			o[oi++] = ex[i];
		}
		i++;
	}
	return calculate(x, xi, o, oi);
}
int main(){
	freopen("in.txt", "r", stdin);
	cin.getline(ti, sizeof(ti));
	cin >> size;
	int i;
	for (i = 0; i < test; i++)
	{
		result[i] = go(ti, i);
	}
	char choose[51];
	cin.getline(choose, 55);
	for (i = 0; i < size; i++){
		cin.getline(choose, 55);
		int j;
		int ans;
		for (j = 0; j < test; j++){
			ans = go(choose, j);
			if (ans != result[j])break;
		}
		if (j == test){
			cout << (char)(i + 'A');
		}
	}
	return 0;
}

vijos-1003等价表达式的更多相关文章

  1. Vijos P1003 等价表达式 随机数+单调栈

    题目链接:https://vijos.org/p/1003 题意: 1. 表达式只可能包含一个变量‘a’. 2. 表达式中出现的数都是正整数,而且都小于10000. 3. 表达式中可以包括四种运算‘+ ...

  2. 数据结构--栈 codevs 1107 等价表达式

    codevs 1107 等价表达式 2005年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond     题目描述 Descripti ...

  3. 等价表达式(noip2005)

    3.等价表达式 [问题描述]    兵兵班的同学都喜欢数学这一科目,中秋聚会这天,数学课代表给大家出了个有关代数表达式的选择题.这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也 ...

  4. 洛谷 P1054 等价表达式 解题报告

    P1054 等价表达式 题目描述 明明进了中学之后,学到了代数表达式.有一天,他碰到一个很麻烦的选择题.这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也是一个代数表达式,题目的 ...

  5. 洛谷P1054 等价表达式

    P1054 等价表达式 题目描述 明明进了中学之后,学到了代数表达式.有一天,他碰到一个很麻烦的选择题.这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也是一个代数表达式,题目的 ...

  6. 等价表达式 2005年NOIP全国联赛提高组(栈模拟)

    P1054 等价表达式 题目描述 明明进了中学之后,学到了代数表达式.有一天,他碰到一个很麻烦的选择题.这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也是一个代数表达式,题目的 ...

  7. 洛谷 P1054 等价表达式

    洛谷 P1054 等价表达式 题目描述 明明进了中学之后,学到了代数表达式.有一天,他碰到一个很麻烦的选择题.这个题目的题干中首先给出了一个代数表达式,然后列出了若干选项,每个选项也是一个代数表达式, ...

  8. 表达式求值(noip2015等价表达式)

    题目大意 给一个含字母a的表达式,求n个选项中表达式跟一开始那个等价的有哪些 做法 模拟一个多项式显然难以实现那么我们高兴的找一些素数代入表达式,再随便找一个素数做模表达式求值优先级表 - ( ) + ...

  9. NOIP2005 等价表达式

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

随机推荐

  1. Mac下开启FTPserver

    开启命令 sudo -s launchctl load -w /System/Library/LaunchDaemons/ftp.plist   关闭命令 sudo -s launchctl unlo ...

  2. 聚合及UML表示

     聚合聚合是一种特别类型的关联,用于描述“总体到局部”的关系. 聚合分成: 基本聚合与合成聚合   基本聚合: 基本聚合一般也简称为聚合(Aggregation).在基本的聚合关系中, 部分类(B)  ...

  3. corefile 设置

    程序运行的过程中,可能会因为一些隐藏的bug导致崩溃,为了在出问题时,及时记录所在环境的情况,所以要设置core文件的产生.其实其本质就是把进程的内存保存到文件中去. 1.core文件的生成开关和大小 ...

  4. html5新增及删除标签

    一.新增标签 有一种划分为,功能性标签[html5新增,如canvas,旧浏览器没有]和语义性标签[如header等只是增强语义,没有新功能].下面按照分几个小类来说. 1.结构标签 新增的结构标签, ...

  5. hdu-5492 Find a path(dp)

    题目链接: Find a path Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  6. UVALive 5000 Underwater Snipers --二分

    题意:一条河岸线y=k,y>k区域有n个敌人,现在要在y<=k区域布置S个狙击手,狙击手的狙击范围为距离自己半径为D的圆内,问满足能够狙死所有的敌人的情况下,离河岸线最近的那个狙击手的离河 ...

  7. Codeforces Round #286 Div.1 A Mr. Kitayuta, the Treasure Hunter --DP

    题意:0~30000有30001个地方,每个地方有一个或多个金币,第一步走到了d,步长为d,以后走的步长可以是上次步长+1,-1或不变,走到某个地方可以收集那个地方的财富,现在问走出去(>300 ...

  8. iOS多线程开发资源抢夺和线程间的通讯问题

    说到多线程就不得不提多线程中的锁机制,多线程操作过程中往往多个线程是并发执行的,同一个资源可能被多个线程同时访问,造成资源抢夺,这个过程中如果没有锁机制往往会造成重大问题.举例来说,每年春节都是一票难 ...

  9. 关于iOS9,Xcode7以上的安全性问题

    目前伴随着苹果方面对安全性方面的重视,在Xcode开发过程中有时候会出现数据解析在view上不显示的问题 这是在iOS9,Xcode7以后苹果方面为了保护用户安全而采用的用户发送请求机制,那么在开发中 ...

  10. ASP.NET MVC验证 - 自定义验证规则、验证2个属性值不等【待验证】

    提示:保存后才提示错误信息 自定义验证特性,继承ValidationAttribute并实现IClientValidatable 这次重写了基类的IsValid()方法的另外一个重载,因为该重载包含了 ...