扩展欧几里得(exgcd)-求解不定方程/求逆元
贝祖定理:
即如果a、b是整数,那么一定存在整数x、y使得ax+by=gcd(a,b)。
换句话说,如果ax+by=m有解,那么m一定是gcd(a,b)的若干倍。(可以来判断一个这样的式子有没有解)
有一个直接的应用就是 如果ax+by=1有解,那么gcd(a,b)=1;
    int gcd(int a,int b){return b==0?a:gcd(b,a%b);}
然而这并不能告诉我们x,y解是多少。
扩欧
首先我们观察上面的式子发现一定有一个解a*1+b*0=gcd(a,b)。(b%a=0)
但是这个ab并不是我们想要的。不过我们可以倒推,原式即:
b*x1+(a%b)*y1=gcd
我们知道:a%b=a-(a/b)*b;带入:
b*x1 + (a-(a/b)*b)*y1
= b*x1 + a*y1 – (a/b)*b*y1
= a*y1 + b*(x1 – a/b*y1) = gcd
发现 x = y1 , y = x1 – a/b*y1
#include<iostream>
#include<cstdio>
using namespace std;
typedef long long ll;
ll exgcd(ll a,ll b,ll &x,ll &y)//这个是返回值版的
{
if(!b)
{
x=1;y=0;
return a; //边界
}
int r=exgcd(b,a%b,x,y);
int temp=y; //把x y变成上一层的
y=x-(a/b)*y;
x=temp; //更改x和y的值,因为是引用的
return r; //得到a b的最大公因数
}
void exgcd1(ll a,ll b,ll &d,ll &x,ll &y)//这个不返回直接更新
{
if(!b)d=a,x=1,y=0;
else
{
exgcd1(b,a%b,d,y,x);
y-=x*(a/b);
}
}
int main()
{
ll a,b,d,x,y;
cin>>a>>b;
d=exgcd(a,b,x,y);//最大公约数
// exgcd1(a,b,d,x,y);
printf("%d %d",x,y);//x,y已经被更新了
return 0;
}
解释一下exgcd1的递推式。
d是最大公约数,这个在b=0,即上一层a%b=0时找到最大公约数,更新传回即可。
我们x和y都是直接更新地址的,那我们这里的y其实就是传给上一层的x,相当于y=x-(a/b)*y,
因为我们传下来的时候是将x和y颠倒的,那么这里x就和y互换,即y-=(a/b)*x,x还是x,传上去就变成y了。
x即为a%b意义下的逆元。
注意,一个数mod一个数是有可能没有逆元的,这时候x好像是返回1。moreover,如果是有逆元的,解出来的x有可能是负数,这时候你可能需要x=(x+mod)%mod.
求不定方程
(我之前没用Markdown写得真难受)
(这个部分是我学了excrt之后才更新的)
(之前学的真是肤浅而且一团乱麻)
exgcd既然可以求ax+by=gcd的解,那么一定就可以求ax+by=c的解。
假设gcd=gcd(a,b),那么我们先把方程两边同时乘gcd/c,那么是不是就可以求x和y了呢?那么求完之后我们在给他乘回去不就行了?
那就行了。
判断无解的情况:如果求出来的gcd并不能被c整除,说明在数论范围内它无解。
鉴于不定方程的性质,x的最小正整数解是在b意义下取模的,y的最小正整数解是在a意义下取模的。我们就可以求出x,y的最小整数解和整数解的个数。因为x和y的增减性是相反的,我们也就可以相应算出最大值。
#include<bits/stdc++.h>
#define int long long //watch out!
using namespace std;
inline int read()
{
int x=0,w=0;char c=getchar();
while(!isdigit(c))w|=c=='-',c=getchar();
while(isdigit(c))x=(x<<3)+(x<<1)+(c^48),c=getchar();
return w?-x:x;
}
inline void write(int x)
{
if(x>9)write(x/10);
putchar(x%10+'0');
}
inline void exgcd(int a,int b,int& d,int& x,int& y){
if(!b)d=a,x=1,y=0;
else exgcd(b,a%b,d,y,x),y-=(a/b)*x;
}
inline void solve(int a,int b,int c)
{
int cnt=0,miny,maxy,minx,maxx;
int gcd,x,y;
exgcd(a,b,gcd,x,y);
if(c%gcd!=0){printf("-1\n");return;}
a/=gcd;b/=gcd;c/=gcd;//这里学习了一下大佬的做法,将所有答案除以gcd,实际上是一样的,不过下面的x和y就不是乘c/gcd了。
x*=c;
y*=c;
minx=x>0&&x%b!=0?x%b:x%b+b;
maxy=(c-minx*a)/b;
miny=y>0&&y%a!=0?y%a:y%a+a;
maxx=(c-miny*b)/a;
if(maxx>0)cnt=(maxx-minx)/b+1;
if(cnt==0)printf("%d %d\n",minx,miny);
else printf("%d %d %d %d %d\n",cnt,minx,miny,maxx,maxy);
}
signed main()
{
int t=read();
while(t--){
int a=read(),b=read(),c=read();
solve(a,b,c);
}
return 0;
}
扩展欧几里得(exgcd)-求解不定方程/求逆元的更多相关文章
- 同余问题(一)——扩展欧几里得exgcd
		
前言 扩展欧几里得算法是一个很好的解决同余问题的算法,非常实用. 欧几里得算法 简介 欧几里得算法,又称辗转相除法. 主要用途 求最大公因数\(gcd\). 公式 \(gcd(a,b)=gcd(b,a ...
 - 浅谈扩展欧几里得[exgcd] By cellur925
		
关于扩展欧几里得从寒假时就很迷,抄题解过了同余方程,但是原理并不理解. 今天终于把坑填上了qwq. 由于本人太菜,不会用markdown,所以这篇总结是手写的(什么).(字丑不要嫌弃嘛) ****** ...
 - poj 1061       扩展欧几里得解同余方程(求最小非负整数解)
		
题目可以转化成求关于t的同余方程的最小非负数解: x+m*t≡y+n*t (mod L) 该方程又可以转化成: k*L+(n-m)*t=x-y 利用扩展欧几里得可以解决这个问题: eg:对于方程ax+ ...
 - 扩展欧几里得(exgcd)与同余详解
		
exgcd入门以及同余基础 gcd,欧几里得的智慧结晶,信息竞赛的重要算法,数论的...(编不下去了 讲exgcd之前,我们先普及一下同余的性质: 若,那么 若,,且p1,p2互质, 有了这三个式子, ...
 - 数论--扩展欧几里得exgcd
		
算法思想 我们想求得一组\(x,y\)使得 \(ax+by = \gcd(a,b)\) 根据 \(\gcd(a,b) = \gcd(b,a\bmod b)\) 如果我们现在有\(x',y'\) 使得 ...
 - 扩展欧几里得 exGCD
		
Elementary Number Theory - Extended Euclid Algorithm Time Limit : 1 sec, Memory Limit : 65536 KB Jap ...
 - 扩展欧几里得,解线性同余方程  逆元 poj1845
		
定理:对于任意整数a,b存在一堆整数x,y,满足ax+by=gcd(a,b) int exgcd(int a,int b,int &x,int &y){ ){x=,y=;return ...
 - 扩展欧几里得 求ax+by == n的非负整数解个数
		
求解形如ax+by == n (a,b已知)的方程的非负整数解个数时,需要用到扩展欧几里得定理,先求出最小的x的值,然后通过处理剩下的区间长度即可得到答案. 放出模板: ll gcd(ll a, ll ...
 - (拓展欧几里得)51NOD 1256 乘法逆元
		
给出2个数M和N(M < N),且M与N互质,找出一个数K满足0 < K < N且K * M % N = 1,如果有多个满足条件的,输出最小的. 输入 输入2个数M, N中间用空 ...
 
随机推荐
- servlet、过滤器、监听器、拦截器之间的关系和区别
			
一.概念 1.什么是servlet servlet是一个接口.定义了一套处理网络请求的规范,所有实现servlet的类,都需要实现它那五个方法,其中最主要的是两个生命周期方法 init()和destr ...
 - 重新整理 .net core 实践篇—————文件系统[二十二]
			
前言 简单介绍一下文件系统. 正文 文件系统,主要是下面3个接口组成: IFileProvider IFileInfo IDirectoryContents 那么他们的实现是: physicalFil ...
 - Nebula 基于 ElasticSearch 的全文搜索引擎的文本搜索
			
本文首发于 Nebula Graph 公众号 NebulaGraphCommunity,Follow 看大厂图数据库技术实践. 1 背景 Nebula 2.0 中已经支持了基于外部全文搜索引擎的文本查 ...
 - 【HTML】同页面锚点跳转
			
跳转: <a href="#maodian001">去吧!</a> 锚点: <a id="maodian001"></ ...
 - k8s service不能访问排错
			
简介 对于新安装的 Kubernetes,经常出现的一个问题是 Service 没有正常工作.如果您已经运行了 Deployment 并创建了一个 Service,但是当您尝试访问它时没有得到响应,希 ...
 - 「模拟8.21」山洞(矩阵优化DP)
			
暴力: 正解: 考虑循环矩阵,f[i][j]表示从i点到j点的方案数 我们发现n很小,我们预处理出n次的f[i][j] 然后在矩阵快速幂中,我们要从当前的f[i][j]*f[j][k]-->fi ...
 - Jenkins+Github+Nginx实现前端项目自动部署
			
前言 最近在搭建一个自己的网站,网站框架搭好了要把项目放到服务器运行,但是每次更新网站内容就要手动部署一次,实在很麻烦,于是就想搭建一套自动化部署的服务.看了一些案例最后选用现在比较主流的Jenkin ...
 - 基于 electron 实现简单易用的抓包、mock 工具
			
背景 经常我们要去看一些页面所发出的请求时,经常会用到 Charles 做为抓包工具来进行接口抓取,但一方面市面是很多抓包工具都是收费或者无法二次开发的.当前我们团队大多数用的也都是 Charles, ...
 - Redis 底层数据结构之字典
			
文章参考 <Redis 设计与实现>黄建宏 字典 在字典中,每个键都是独一无二的,程序可以在字典中根据键查找与之相关联的值,或者通过键来更新和删除值. 字典在 Redis 中的应用相当广泛 ...
 - 12.5finally子句
			
要点提示:无论异常是否产生,finally子句总是会执行的. 有时候无论异常是否出现或者是否被捕获,都希望执行某些代码.java有一个finally子句,可以用来达到这个目的. 注意:使用finall ...