poj 2891 扩展欧几里得迭代解同余方程组
Reference: http://www.cnblogs.com/ka200812/archive/2011/09/02/2164404.html
之前说过中国剩余定理传统解法的条件是m[i]两两互质,所以这题就不能用传统解法了= =
其实还有种方法:
先来看只有两个式子的方程组:
c≡b1 (mod a1)
c≡b2 (mod a2)
变形得c=a1*x+b1,c=a2*x+b2
a1*x-a2*y=b2-b1
可以用扩展欧几里得求出x和y,进而求出c
那么多个式子呢?可以两个两个的迭代求。
比如上面两个式子求完了,求出一个c,不妨先把它记作c1
c1满足上面两式,但对于所有的式子就不一定都满足了。
因此我们把一个新同余式加入方程组:c≡c1 (mod lcm(a1,a2)) //lcm为最小公倍数
然后依次往下迭代就行了。最后解出的c1就是最终解。
如何判断无解?
对于相邻的两行a1、a2、r1、r2,若 (r2-r1) mod (gcd(a1,a2))!=0说明无解
#include "iostream"
using namespace std;
__int64 a[];
__int64 r[];
int n; __int64 extend_gcd(__int64 a,__int64 b,__int64 &x,__int64 &y){
if (b==){
x=;y=;
return a;
}
else{
__int64 r=extend_gcd(b,a%b,y,x);
y=y-x*(a/b);
return r;
}
} __int64 gcd(__int64 a,__int64 b)
{
if (b==) return a;
return gcd(b,a%b);
} __int64 lcm(__int64 a,__int64 b)
{
__int64 tm=gcd(a,b);
if (tm==) return ;
else return (a*b/tm);
} int main()
{
while (cin>>n)
{
for (int i=;i<=n;i++)
cin>>a[i]>>r[i]; __int64 x,y,x1,c1;
bool sol=true;
for (int i=;i<=n;i++)
{
__int64 a1=a[i-],r1=r[i-];
__int64 a2=a[i],r2=r[i];
__int64 tmp=gcd(a1,a2);
if ((r2-r1)%tmp!=) sol=false; //此处有个问题= =原来我是这样写的,WA了
__int64 tm=extend_gcd(a1,a2,x,y); //__int64 tm=extend_gcd(a1,-a2,x,y)
x1=x*((r2-r1)/tm); //but the original equation is a1*x1-a2*x2=b2-b1
__int64 rq=a2/tm; //__int64 rq=-a2/tm
x1=(x1%rq+rq)%rq; //去掉a2的负号就A了,What a fuck!!!
//个人猜想是因为gcd(a,b)和gcd(a,-b)结果是一样的,
//而且此处需要的是非负解,结果就YY对了
c1=a1*x1+r1;
r[i]=c1;
a[i]=lcm(a1,a2);
// cout<<r[i]<<" "<<a[i]<<endl;
}
//cout<<c1<<endl;
__int64 ans=c1;
if (!sol) cout<<"-1"<<endl;
else cout<<ans<<endl;
} return ;
}
至于负号那个地方有个题hdu 1576,和本题一个情况。把负号YY掉就对了= =
poj 2891 扩展欧几里得迭代解同余方程组的更多相关文章
- POJ 2891 Strange Way to Express Integers | exGcd解同余方程组
题面就是让你解同余方程组(模数不互质) 题解: 先考虑一下两个方程 x=r1 mod(m1) x=r2 mod (m2) 去掉mod x=r1+m1y1 ......1 x=r2+m2y2 . ...
- poj 2142 扩展欧几里得解ax+by=c
原题实际上就是求方程a*x+b*y=d的一个特解,要求这个特解满足|x|+|y|最小 套模式+一点YY就行了 总结一下这类问题的解法: 对于方程ax+by=c 设tm=gcd(a,b) 先用扩展欧几里 ...
- poj 1061 扩展欧几里得解同余方程(求最小非负整数解)
题目可以转化成求关于t的同余方程的最小非负数解: x+m*t≡y+n*t (mod L) 该方程又可以转化成: k*L+(n-m)*t=x-y 利用扩展欧几里得可以解决这个问题: eg:对于方程ax+ ...
- poj 1061(扩展欧几里得定理求不定方程)
两只青蛙在网上相识了,它们聊得很开心,于是觉得很有必要见一面.它们很高兴地发现它们住在同一条纬度线上,于是它们约定各自朝西跳,直到碰面为止.可是它们出发之前忘记了一件很重要的事情,既没有问清楚对方的特 ...
- 扩展欧几里得(exgcd)与同余详解
exgcd入门以及同余基础 gcd,欧几里得的智慧结晶,信息竞赛的重要算法,数论的...(编不下去了 讲exgcd之前,我们先普及一下同余的性质: 若,那么 若,,且p1,p2互质, 有了这三个式子, ...
- poj 2115 扩展欧几里得
题目链接:http://poj.org/problem?id=2115 题意: 给出一段循环程序,循环体变量初始值为 a,结束不等于 b ,步长为 c,看要循环多少次,其中运算限制在 k位:死循环输出 ...
- The Balance POJ 2142 扩展欧几里得
Description Ms. Iyo Kiffa-Australis has a balance and only two kinds of weights to measure a dose of ...
- POJ 1061 扩展欧几里得
#include<stdio.h> #include<string.h> typedef long long ll; void gcd(ll a,ll b,ll& d, ...
- 扩展欧几里得,解线性同余方程 逆元 poj1845
定理:对于任意整数a,b存在一堆整数x,y,满足ax+by=gcd(a,b) int exgcd(int a,int b,int &x,int &y){ ){x=,y=;return ...
随机推荐
- flex4 s:Datagrid <s:typicalItem
<s:DataGird <s:typicalItem 这个标签相信大家很陌生吧, 我也是今天准备讲的时候才看到,估计是 flex4.5.1 新加东西,果然摸索 了下,这个标签作用也蛮好用的 ...
- JetBrain WebStorm 注册码
webStorm : UserName:William ===== LICENSE BEGIN ===== 45550-12042010 00001SzFN0n1bPII7FnAxnt0DDOPJA ...
- 实现了一个简单的cage变形器
今天实现了一个简单变形器,可以用一个网格的形状影响另一个网格的形状. 如图,蓝色网格的形状被灰色网格操控. 当前的算法非常简单,就是计算蓝色网格每个点到灰色网格每个点的距离,以距离x次方的倒数作为权重 ...
- 用运算符代替if、else
- CLR执行模式之程序集代码的执行
所知IL是与CPU无关的机器语言,其能访问和操作对象类型,并提供指令来创建和初始化对象,调用对象上的虚方法以及直接操作数组对象等,故可视为一种面向对象的机器语言.每种语言的存在都有其存在的价值和原因, ...
- SSH公钥认证+优化
一 ssh公钥认证流程: sshclinet机器:产生公私钥(公钥相当于一把锁) sshclient:将公钥发给sshserver(抛出锁子) sshclinet去连sshserver不需要密钥 ...
- Android 下的EXIF
一.什么是Exif Exif(Exchangeable Image File 可交换图像文件)是一种图象文件格式,它的数据存储与JPEG格式是完全相同的.实际上Exif格式就是在JPEG格式头部插入了 ...
- C语言 文件操作2--文件缓存的理解
//文件缓存机制理解 #define _CRT_SECURE_NO_WARNINGS #include<stdio.h> #include<stdlib.h> void mai ...
- MVC4验证用户登录特性实现方法
在开发过程中,需要用户登陆才能访问指定的页面这种功能,微软已经提供了这个特性. // 摘要: // 表示一个特性,该特性用于限制调用方对操作方法的访问. [AttributeUsage(Attribu ...
- http 请求头设置缓存
nginx不缓存设置 2013-08-15 10:47:39 分类: LINUX 在开发调试web的时候,经常会碰到因浏览器缓存(cache)而经常要去清空缓存或者强制刷新来测试的烦恼,提供下apa ...