UVA 12169 Disgruntled Judge 枚举+扩展欧几里得
2018-06-17 23:35:18来源:未知 阅读 ()
题目大意:有3个整数 x[1], a, b 满足递推式x[i]=(a*x[i-1]+b)mod 10001。由这个递推式计算出了长度为2T的数列,现在要求输入x[1],x[3],......x[2T-1], 输出x[2],x[4]......x[2T]. T<=100,0<=x<=10000. 如果有多种可能的输出,任意输出一个结果即可。
由于a和b都小于等于10000,直接枚举a和b暴力可以过。但是有没有更快的方法呢?
首先令递推式的i=2,那么x[2]=(a*x[1]+b)mod 10001;再令i=3,得x[3]=(a*x[2]+b)mod 10001,可以得出x[3]=(a*(a*x[1]+b)+b)mod 10001。这时候只有a和b是变量,我们枚举a,就可以求出b了。(a+1)*b mod 10001 = ( (x[3]-a*a*x[1]) mod 10001 + 10001 ) mod 10001.(这里的x[3]-a*a*x[1]可能为负,代码中可以先不取模,后面计算b的时候一起取模即可) 所以简化成(a+1)*b mod 10001 = (x[3]-a*a*x[1]) mod 10001。这里就变成了同模方程,扩展欧几里得即可解答。
暴力代码:
#include <iostream> #include <cstdio> #include <cstring> using namespace std; const int maxn=10000+5; const int mod=10001; int in[maxn]; int main() { //freopen("in.txt","r",stdin); int t; scanf("%d",&t); for(int i=0; i<t; i++) scanf("%d",in+i); bool flag; for(int a=0; a<=10000; a++) { for(int b=0; b<=10000; b++) { flag=false; for(int i=1; i<t; i++) if(in[i]!=((a*(a*in[i-1]%mod+b)+b)%mod)) { flag=true; break; } if(!flag) { for(int i=0; i<t; i++) printf("%d\n",(a*in[i]+b)%mod); break; } } if(!flag) break; } return 0; }
扩展欧几里得:
#include <iostream> #include <cstdio> #include <cstring> using namespace std; const int maxn=10000+5; const int mod=10001; int in[maxn]; typedef long long ll; ll exgcd(ll a, ll b, ll&x, ll&y) { if (b == 0) { x = 1; y = 0; return a; } ll r = exgcd(b, a%b, y, x); ll t = x; y = y - a/b*t; return r; } int main() { //freopen("in.txt","r",stdin); int t; scanf("%d",&t); for(int i=0; i<t; i++) scanf("%d",in+i); bool flag; for(ll a=0; a<=10000; a++) { ll x,y; //定义long long 型是保证没有取模的式子不会超内存 ll g=exgcd(a+1,mod,x,y); ll tmp=in[1]-a*a*in[0]; //这里可以先不取模,后面计算b的时候取模 if(tmp%g==0) { flag=false; ll b=(x*tmp/g)%mod; //这里最好取下模,虽然后面计算in[i]的时候也会取模,但是算出来的in[i]可能因为b负太多而变成负数 for(int i=1;i<t;i++) { if(in[i]!=(a*(a*in[i-1]+b)+b)%mod) { flag=true; break; } } if(!flag) { for(int i=0;i<t;i++) printf("%d\n",(a*in[i]+b)%mod); break; } } } return 0; }
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
- [Uva1637][DFS][记忆化] 纸牌游戏 Double Patience 2020-03-06
- Prime Time UVA - 10200(精度处理,素数判定) 2019-08-16
- J - Fire! UVA - 11624 2018-09-01
- uva11768 Lattice Point or Not 2018-08-14
- UVALive - 6434 (贪心) 2018-07-28
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash