Re:Exgcd解二元不定方程
模拟又炸了,我死亡
$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$。
#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 ;
}
解二元不定方程
设跳了$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解二元不定方程的更多相关文章
- ecgcd(解二元不定方程)
题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=775 关于扩展欧几里得算法还是推一遍好啦: 有方程:a*x+b*y=d=gcd(a, b) ...
- 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 ...
- Java实现 蓝桥杯VIP 算法提高 解二元一次方程组
算法提高 解二元一次方程组 时间限制:1.0s 内存限制:256.0MB 问题描述 给定一个二元一次方程组,形如: a * x + b * y = c; d * x + e * y = f; x,y代 ...
- 使用代数方程库 Algebra.js解二元一次方程
假设二元一次方程如下: x + y = 11 x - y = 5 解方程如下: <!DOCTYPE html> <html lang="zh-CN"> &l ...
- POJ 2891 Strange Way to Express Integers | exGcd解同余方程组
题面就是让你解同余方程组(模数不互质) 题解: 先考虑一下两个方程 x=r1 mod(m1) x=r2 mod (m2) 去掉mod x=r1+m1y1 ......1 x=r2+m2y2 . ...
- HDU 4793 Collision (解二元一次方程) -2013 ICPC长沙赛区现场赛
题目链接 题目大意 :有一个圆硬币半径为r,初始位置为x,y,速度矢量为vx,vy,有一个圆形区域(圆心在原点)半径为R,还有一个圆盘(圆心在原点)半径为Rm (Rm < R),圆盘固定不动,硬 ...
- nyoj 64-鸡兔同笼 (解二元一次方程)
64-鸡兔同笼 内存限制:64MB 时间限制:3000ms Special Judge: No accepted:26 submit:58 题目描述: 已知鸡和兔的总数量为n,总腿数为m.输入n和m, ...
- c语言解二元二次方程组
设a和b是正整数 a+b=30 且a*b=221 求a和b的值 思路就是穷举a和b的值,每次得到a和b的一个值,看是否同时满足a+b=30且a*b=221,如果满足,那么就输出. 那么a和b的的取值范 ...
- VBA Promming——分支语句(解二元一次方程)
分支语句 If expression1 Then expressions ElseIf expression2 Then expressions Else expression End If 注:VB ...
随机推荐
- linux下(fdisk,gdisk,parted)三种分区工具比较
1 2种分区结构简介 MBR分区 硬盘主引导记录MBR由4个部分组成 主引导程序(偏移地址0000H--0088H),它负责从活动分区中装载,并运行系统引导程序. 出错信息数据区,偏移地址0089H- ...
- Spring Boot(五):Spring Boot Jpa 的使用
在上篇文章Spring Boot(二):Web 综合开发中简单介绍了一下 Spring Boot Jpa 的基础性使用,这篇文章将更加全面的介绍 Spring Boot Jpa 常见用法以及注意事项. ...
- Django学习笔记(1)——初识Django
一:Web框架介绍 框架,即framework,特指为解决一个开放性问题而设计的具有一定约束性的支撑结构,使用框架可以快速帮你开发特定的系统. Web框架是别人已经设定好的一个web网站模板,你学习它 ...
- 线程的私有领地 ThreadLocal
从名字上看,『ThreadLocal』可能会给你一种本地线程的概念印象,可能会让你联想到它是一个特殊的线程. 但实际上,『ThreadLocal』却营造了一种「线程本地变量」的概念,也就是说,同一个变 ...
- ubuntu开发项目不能执行热更新
当项目开发到一定成熟度,项目基本上比较大(vue,angular,react,java,php等),在Ubuntu系统环境下,我们写了代码,但是不能想Windows一样执行热更新,这是因为Ubuntu ...
- linux yum配置代理
yum里面可以单独设置代理就是yum源的参数加proxy=“http://ip:PORT”即在/etc/yum.conf中加入下面几句.proxy=http://210.45.72.XX:808pro ...
- 学JAVA的第二天,静态网站制作,脑阔一点疼
先从下载apache-tomcat-9.0.17开始 在下边这个网站下载,下边一步步来 下面删除的这些是暂时用不上的,先吧它删除了,因为会拖慢启动速度 下边把ROOT里边除WEB-INF外的全不删除了 ...
- Java开发笔记(六十六)映射:HashMap和TreeMap
前面介绍了两种集合的用法,它们的共性为每个元素都是唯一的,区别在于一个无序一个有序.虽说往集合里面保存数据还算容易,但要从集合中取出数据就没那么方便了,因为集合居然不提供get方法,没有get方法怎么 ...
- Catalan卡特兰数入门
简介 卡特兰数是组合数学中的一种常见数列 它的前几项为: 1, 1, 2, 5, 14, 42, 132, 429, 1430, 4862, 16796, 58786, 208012, 742900, ...
- 【20190405】JavaScript-整理一些常用正则式
匹配中文字符: let reg=/([\u4E00-\u9FFF]+)/; //\u代表Unicode编码 匹配电话号码: let reg=/^1[34578]\d{9}$/; 给每三位数字添加一个逗 ...