模拟又炸了,我死亡

$exgcd$(扩展欧几里德算法)用于求$ax+by=gcd(a,b)$中$x,y$的一组解,
它有很多应用,比如解二元不定方程、求逆元等等,
这里详细讲解一下$exgcd$的原理。

了解$exgcd$算法前,需要$gcd$算法做铺垫。gcd,又称辗转相除法,用于计算两个整数 $a,b$ 的最大公约数。

$gcd$函数的基本性质:

$gcd(a,b)=gcd(b,a) $

$gcd(a,b)=gcd(-a,b) $

$gcd(a,b)=gcd(|a|,|b|)$

$gcd(a,b)=gcd(b,a$ $mod$ $b)$

证明一下第四条:

设$k$为整数

设$gcd(a,b) = d$

$a$ $mod$ $b=a-kb$,$k = a/b$

因为$a$ $mod$ $d=0,b$ $mod$ $d=0$

所以$kb$ $mod$ $d=0$,

$a-kb$ $mod$ $d=0$

即$(a$ $mod$ $b)$ $mod$ $d=0$

int gcd(int a,int b){
if(b == ){
return a;
}
return gcd(b,a%b);
}

  gcd  

理解了$gcd$,再来证明为什么$exgcd$可以求出$ax+by=gcd(a,b)$的$x,y$。

当$b=0$时,$gcd(a,b) = a$,$ax+by$ 即 $a*1 + 0*0 = gcd(a,b)$;

$x=1,y=0$(因为$b = 0$,$y$的值其实不重要,这里就写成$0$)

因为$gcd(a,b) =  ax+by$,$gcd(b,a$ $mod$ $b) = ax'+ by'$,

每次递归时,$gcd(a,b) = gcd(b,a$ $mod$ $b)$,所以$ax+by = ax'+ by'$,

并且已知 $a%b = a-(a/b)*b$,那么可以得到

$ax'+ by'$

$= bx + (a$ $mod$ $b)y $

$= bx + (a-a/b*b)y $

$= bx + ay - (a/b)*by $

$= ay + b(x-(a/b)*y) $

$x'=y$; $y'=x-(a/b)*y $

int exgcd(int a,int b,int &x, int &y) {
if(b == ) {
x=,y=;
return a;
}
int ans = exgcd(b,a%b,x,y);
int tx = x;
x = y;
y = tx-a/b*y;
return ans;
}

  exgcd  

明白了$exgcd$的原理,思考它如何求解形如$ax+by=c$的不定方程。

设$k$为整数,

设$c=k*gcd(a,b)$

$ax+by=gcd(a,b)$,

两边同乘$k$,得

$k*ax+k*by=k*gcd(a,b)=c$

所以$c$一定为$gcd(a,b)$的倍数。若$c$ $mod$ $gcd(a,b) ≠ 0$,则方程无整数解。

此时的$x'=kx=c/gcd(a.b)*x$。

可以发现,$ax+by=c$ (x∈Z*,y∈Z*)  成立的充要条件是$gcd(a,b)|c$。这个定理叫做裴蜀定理
#include<cstdio>
#include<iostream>
#include<cmath>
#include<cstring>
#define MogeKo qwq
using namespace std; int n,x,sum; int gcd(int a,int b) {
if(!b) return a;
return gcd(b,a%b);
} int main() {
scanf("%d",&n);
sum = ;
for(int i = ; i <= n; i++) {
scanf("%d",&x);
if(x < ) x = -x;
sum = gcd(sum,x);
}
printf("%d",sum);
return ;
}

裴蜀定理

那么,怎么保证x是最小正整数呢?

设$k$为整数,   

   $ax+by$

  $= ax + by + k*ab - k*ab$

  $= a(x+kb) + b(y-ka)$

也就是说,当$x$改变$b$的整数倍时,原式的值可以保持不变;

那么最小正整数即为$x$ $mod$ $b$;

但是要考虑$x$为负数的情况,就要先将$x$加上$b$,直到$x$为正数,再取模,可以得到$x = (x$ $mod$ $b+b)$ $mod$ $b$

综上所述,

$x=(x*c/gcd(a,b)$ $mod$ $b+b)$ $mod$ $b$

int exgcd(int a,int b,int &x, int &y) {
if(b == ) {
x=,y=;
return a;
}
int ans = exgcd(b,a%b,x,y);
int tx = x;
x = y;
y = tx-a/b*y;
return ans;
} int main() {
scanf("%d%d%d",&a,&b,&c);
if(a<) {
a = -a;
c = -c;
}
exgcd(a,b,x,y);
if(c%ans!=)
printf("Impossible");
else
printf("%d",(x*(c/ans))%(b/ans)+(b/ans))%(b/ans));
return ;
}

  解二元不定方程 

模板题:Luogu P1516 青蛙的约会

设跳了$X$次,

由题意得,$m*X+x=n*X+y$ $(mod$ $l)$

$(m-n)*X=y-x$ $(mod$ $l)$

$(m-n)*X-l*Y=y-x$

令$m-n=a$,$l=b$,$y-x=c$,$Y=-Y$

则有$aX+bY=c$

注意负数的情况:若$m-n<0$,则变成了前面的追后面的

$a=n-m$,$c=(x+l-y)$ $mod$ $l=x-y$

即$a=-a,c=-c$

#include<cstdio>
#define int long long
int x,y,m,n,l,a,b,c,x0,y0,g,tmp; int exgcd(int a,int b,int &x, int &y) {
if(b == ) {
x=,y=;
return a;
}
int ans = exgcd(b,a%b,x,y);
int tx = x;
x = y;
y = tx-a/b*y;
return ans;
} main() {
scanf("%lld%lld%lld%lld%lld",&x,&y,&m,&n,&l);
a=n-m;
b=l;
c=x-y;
if(a<)
a=-a,c=-c;
int g = exgcd(a,b,x0,y0);
if(c%g!=)
printf("Impossible");
else
printf("%lld",(c/g*x0%(b/g)+b/g)%(b/g));
}
												

Re:Exgcd解二元不定方程的更多相关文章

  1. ecgcd(解二元不定方程)

    题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=775 关于扩展欧几里得算法还是推一遍好啦: 有方程:a*x+b*y=d=gcd(a, b) ...

  2. exgcd 解同余方程ax=b(%n)

    ax=n(%b)  ->   ax+by=n 方程有解当且仅当 gcd(a,b) | n ( n是gcd(a,b)的倍数 ) exgcd解得 a*x0+b*y0=gcd(a,b) 记k=n/gc ...

  3. Java实现 蓝桥杯VIP 算法提高 解二元一次方程组

    算法提高 解二元一次方程组 时间限制:1.0s 内存限制:256.0MB 问题描述 给定一个二元一次方程组,形如: a * x + b * y = c; d * x + e * y = f; x,y代 ...

  4. 使用代数方程库 Algebra.js解二元一次方程

    假设二元一次方程如下: x + y = 11 x - y = 5 解方程如下: <!DOCTYPE html> <html lang="zh-CN"> &l ...

  5. POJ 2891 Strange Way to Express Integers | exGcd解同余方程组

    题面就是让你解同余方程组(模数不互质) 题解: 先考虑一下两个方程 x=r1 mod(m1) x=r2 mod (m2) 去掉mod x=r1+m1y1   ......1 x=r2+m2y2   . ...

  6. HDU 4793 Collision (解二元一次方程) -2013 ICPC长沙赛区现场赛

    题目链接 题目大意 :有一个圆硬币半径为r,初始位置为x,y,速度矢量为vx,vy,有一个圆形区域(圆心在原点)半径为R,还有一个圆盘(圆心在原点)半径为Rm (Rm < R),圆盘固定不动,硬 ...

  7. nyoj 64-鸡兔同笼 (解二元一次方程)

    64-鸡兔同笼 内存限制:64MB 时间限制:3000ms Special Judge: No accepted:26 submit:58 题目描述: 已知鸡和兔的总数量为n,总腿数为m.输入n和m, ...

  8. c语言解二元二次方程组

    设a和b是正整数 a+b=30 且a*b=221 求a和b的值 思路就是穷举a和b的值,每次得到a和b的一个值,看是否同时满足a+b=30且a*b=221,如果满足,那么就输出. 那么a和b的的取值范 ...

  9. VBA Promming——分支语句(解二元一次方程)

    分支语句 If expression1 Then expressions ElseIf expression2 Then expressions Else expression End If 注:VB ...

随机推荐

  1. Angularjs1.X进阶笔记(1)—两种不同的双向数据绑定

    一. html与Controller中的双向数据绑定 html-Controller的双向数据绑定,在开发中非常常见,也是Angularjs1.x的宣传点之一,使用中并没有太多问题. 1.1数据从ht ...

  2. [转]Chrome 错误代码:ERR_UNSAFE_PORT

    本文转自:https://blog.csdn.net/testcs_dn/article/details/39186225 最近在用Nginx发布多个站点测试,使用了87.88端口, 88端口访问正常 ...

  3. 初识 MongoDB,MongoDB 的安装运行

    1.  MongoDB 非关系型数据库  MongoDB是一个基于分布式文件存储的数据库,由C++语言编写.目的是为WEB应用提供扩展的高性能的数据存储解决方案.MongoDB是一个介于关系型数据库和 ...

  4. [MySQL] 索引中的b树索引

    1.索引如果没有特别指明类型,一般是说b树索引,b树索引使用b树数据结构存储数据,实际上很多存储引擎使用的是b+树,每一个叶子节点都包含指向下一个叶子节点的指针,从而方便叶子节点的范围遍历 2.底层的 ...

  5. python爬虫之静态网页——全国空气质量指数(AQI)爬取

    首先爬取地址:http://www.air-level.com/ 利用的python库,最近最流行的requests,BeautifulSoup. requests:用于下载html Beautifu ...

  6. SpringBoot项目部署到服务器上,tomcat不启动该项目

    今天lz把项目重新传到服务器上后,重启tomcat遇到个问题,就是这个tomcat怎么都不启动这个项目,别的项目都没事,一番查找后发现问题所在. 我们先建个SpringBoot工程,重现一下问题: 写 ...

  7. Bootstrap 实战之响应式个人博客 (一)

    一.示例 1.主页 2.博客详情页 3.在线地址 在线地址:入口 Addition:这里使用github-page将自己的静态项目免费部署到线上. 如果你只是做一些简单的静态项目做展示,付出这么大的时 ...

  8. H5移动端rem适配

    /** * 移动端自适应 */ <meta name="viewport" content="width=device-width,user-scalable=no ...

  9. 2019-01-23 JavaScript实现ZLOGO: 性能改进

    主攻前文吴烜:JavaScript实现ZLOGO: 界面改进与速度可调的几个性能问题 在线演示: 圈3 源码仍在: program-in-chinese/quan3 之前是在绘制过程中计算每帧需要绘制 ...

  10. 参观微软Serbia开发中心和Office365 2019-01-31活动感悟

    这是<国外线下技术俱乐部建设>系列文章之一.   该活动网址是:https://www.meetup.com/ITPro-Serbia/events/258352104/ 活动内容是讲Of ...