浅谈扩展欧几里得(扩展GCD)算法

本篇随笔讲解信息学奥林匹克竞赛中数论部分的扩展欧几里得算法。为了更好的阅读本篇随笔,读者最好拥有不低于初中二年级(这是经过慎重考虑所评定的等级)的数学素养。并且已经学会了学习这个算法的前置知识:欧几里得算法。

对于对欧几里得算法还有知识模糊的读者,请不要担心,这里为你准备了前导知识讲解,请移步至本蒟蒻的另两篇博客:

浅谈GCD

求最大公约数的方式

裴蜀定理

裴蜀定理的概念及证明

因为翻译版本的不同,这个定理可能还会被叫做贝祖定理、\(B\acute{e}zout\)定理等。

裴蜀定理是这样被描述的:

\[\forall a,b\in Z\,\,,\,\,\exists (x,y)\in Z
\]

满足:

\[ax+by=gcd(a,b)
\]

文字描述是这样的:对于任意的整数\(a,b\),都存在一对整数\(x,y\),使得\(ax+by=\gcd(a,b)\)成立。

证明:

来看欧几里得算法求解过程的这个函数:

int gcd(int a,int b)
{
return b?gcd(b,a%b):a;
}

可以看出,这是一个递归求解的函数。在函数递归到最后的时候,存在\(b=0\),不管\(a\)是什么,这时显然有一对整数\(x=1,y=0\)来使得:

\[a\times 1+0\times 0=\gcd(a,0)
\]

(ps:0和任何数的最大公约数都等于原数,可以从最大公约数和约数的定义得知)

那么,我们通过这个递归的实现过程来进行回溯的模拟。当\(b>0\),则程序还可以继续往下走:\(\gcd(b,a\%b)=\gcd(a,b)\)。这时假设存在一对整数\(x,y\),使得其一定会满足\(b\times x+(a\%b)\times y=\gcd(b,a\%b)\), 因为\(a\%b=a-b\lfloor a/b\rfloor\),所以有以下的推导:

\[b\times x+(a\%b)\times y=\gcd(b,a\%b)=bx+(a-b\lfloor a/b\rfloor)y=ay-b(x-\lfloor a/b\rfloor y)
\]

这个时候令\(x^{'}=y,y^{'}=x-\lfloor a/b\rfloor y\),再结合一开始的原式子,就得出:

\[ax^{'}+by^{'}=\gcd(a,b)
\]

因为欧几里得算法的实现是递归的,而我们已经推出其中一个递归过程的实现,那么其他的递归过程就可以借助数学归纳法,一层层地向上推,必然会得出最终结论。

证毕。

裴蜀定理的应用

裴蜀定理:

\[ax+by=\gcd(a,b)
\]

那么可以推出:如果一个数\(m\)满足:\(ax+by=m\),那么这个\(m\)一定是\(\gcd(a,b)\)的倍数。

那么对于一个经典方程\(ax+by=1\),利用裴蜀定理,我们有:\(\gcd(a,b)=1\),即\(a,b\)一定互质。


扩展欧几里得算法

在介绍扩展欧几里得算法之前,我想首先介绍它的应用:

1、求解不定方程

2、求解模的逆元

3、求解线性同余方程

为什么它能应用到这几个方面呢?回到裴蜀定理:

\[ax+by=m
\]

对于这个不定方程,如果存在一组合法的解\((x,y)\),那么一定会有\(\gcd(a,b)|m\),即\(m\)是\(\gcd(a,b)\)的倍数。那么现在我不仅想知道到底有没有解,而是想知道在有解的情况下,这个解到底是多少。

这就是求解不定方程的过程。这个解决的算法就叫做扩展欧几里得算法

可以发现,我们求解不定方程其实就是要求解一组合法的\((x,y)\),那么根据裴蜀定理的证明(基于欧几里得算法,采用递归的数学归纳),可以发现\(x,y\)的互相推导的关系。

这种采取递归来求解\(x,y\)的方法就叫做扩展欧几里得算法。

扩展欧几里得算法的实现:

先放板子:

int exgcd(int a,int b,int &x,int &y)
{
if(b==0)
{
x=1,y=0;
return a;
}
int d=exgcd(b,a%b,x,y);
int k=x;
x=y;
y=k-a/b*y;
return d;
}

扩展欧几里得算法的实现基于裴蜀定理的证明。实质上相当于在做欧几里得算法(普通GCD)的时候对不定方程\(ax+by=m\)的\(x,y\)也做了更改。所以经过扩展欧几里得算法处理过的\(x,y\)就已经是一组合法的可行解了。

这里需要注意一个细节,因为扩展GCD需要对\(x,y\)本身进行修改,所以需要在传参数的时候加取址符,这样能保证被修改。

详解扩展欧几里得算法(扩展GCD)的更多相关文章

  1. 初等数论-Base-2(扩展欧几里得算法,同余,线性同余方程,(附:裴蜀定理的证明))

    我们接着上面的欧几里得算法说 扩展欧几里得算法 扩展欧几里德算法是用来在已知a, b求解一组x,y,使它们满足贝祖等式\(^①\): ax+by = gcd(a, b) =d(解一定存在,根据数论中的 ...

  2. 扩展欧几里得算法详解(exgcd)

    一.前言 本博客适合已经学会欧几里得算法的人食用~~~ 二.扩展欧几里得算法 为了更好的理解扩展欧几里得算法,首先你要知道一个叫做贝祖定理的玄学定理: 即如果a.b是整数,那么一定存在整数x.y使得$ ...

  3. 模板——扩展欧几里得算法(求ax+by=gcd的解)

    Bryce1010模板 /**** *扩展欧几里得算法 *返回d=gcd(a,b),和对应等式ax+by=d中的x,y */ long long extend_gcd(long long a,long ...

  4. 【learning】 扩展欧几里得算法(扩展gcd)和乘法逆元

    有这样的问题: 给你两个整数数$(a,b)$,问你整数$x$和$y$分别取多少时,有$ax+by=gcd(x,y)$,其中$gcd(x,y)$表示$x$和$y$的最大公约数. 数据范围$a,b≤10^ ...

  5. gcd(欧几里得算法)与exgcd(扩展欧几里得算法)

    欧几里得算法: 1.定义:gcd的意思是最大公约数,通常用扩展欧几里得算法求 原理:gcd(a, b)=gcd(b, a%b) 2.证明: 令d=gcd(a, b)  =>  a=m*d,b=n ...

  6. 扩展欧几里得算法(extgcd)

    相信大家对欧几里得算法,即辗转相除法不陌生吧. 代码如下: int gcd(int a, int b){ return !b ? gcd(b, a % b) : a; } 而扩展欧几里得算法,顾名思义 ...

  7. 欧几里得算法与扩展欧几里得算法_C++

    先感谢参考文献:http://www.cnblogs.com/frog112111/archive/2012/08/19/2646012.html 注:以下讨论的数均为整数 一.欧几里得算法(重点是证 ...

  8. vijos1009:扩展欧几里得算法

    1009:数论 扩展欧几里得算法 其实自己对扩展欧几里得算法一直很不熟悉...应该是因为之前不太理解的缘故吧这次再次思考,回看了某位大神的推导以及某位大神的模板应该算是有所领悟了 首先根据题意:L1= ...

  9. ****ural 1141. RSA Attack(RSA加密,扩展欧几里得算法)

    1141. RSA Attack Time limit: 1.0 secondMemory limit: 64 MB The RSA problem is the following: given a ...

随机推荐

  1. 控制台提示“Invalid string length”的原因

    控制台提示“Invalid string length”,浏览器直接卡掉,是为什么呢? 答:因为在写嵌套循环时,定义的变量重名了,内层和外层用了同一个i变量. -THE END-

  2. Visual Studio 开发(三):Visual Studio 使用时常见问题解决方案

    一.Error LNK2019: 无法解析的外部符号 此问题应该是Visual Studio的初学者最常碰到的问题,也是相对来说很让人头疼的问题. 注:Error LNK2019 问题在VC 6.0 ...

  3. 3、nio中的selector使用

    通过编写一个客户端和服务器端的例子来熟悉selector的使用 服务端逻辑: 1. 绑定一个端口号2. channel注册到selector中3. 用死循环来监听如果有时间发生,遍历selection ...

  4. [Go] 利用函数类型实现封装中的回调

    当进行业务逻辑开发的时候,经常要进行封装,封装成独立的类文件,在类文件的属性中预留出函数类型的API 在调用该类文件中某些方法的时候,也根据业务需要调用类属性中的函数, 在主业务中可以传递特定的函数注 ...

  5. linux 常用命令及软件

    命令基于ubuntu 18.04 修改网卡配置 /etc/netplan/50-cloud-init.yaml #修改 netplan apply #应用修改 修改计算机名 sudo hostname ...

  6. mybatis入门视频总结

    1.基础知识 2. 原生态jdbc问题总结 3.mybatis是什么 4.工程结构 5.SqlMapConfig.xml文件 6.sqlmap 包下新建user.xml文件(namespace,par ...

  7. C++ std::vector 基本用法2

    #include <iostream> #include <vector> using namespace std; int main() { int ar[10] = { 1 ...

  8. Redis令牌桶限流

    一 .场景描述 在开发接口服务器的过程中,为了防止客户端对于接口的滥用,保护服务器的资源, 通常来说我们会对于服务器上的各种接口进行调用次数的限制.比如对于某个 用户,他在一个时间段(interval ...

  9. [CrackMe]160个CrackMe之015

    吾爱破解专题汇总:[反汇编练习]160个CrackME索引目录1~160建议收藏备用 一.破解 该破解比较简单,其是一个静态密码  2G83G35Hs2 ,输入进去即可破解. 1)栈定位法找到用户代码 ...

  10. 学习强国docker文件用法

    学习强国docker用法 docker文件地址   https://github.com/fuck-xuexiqiangguo/docker 构建  docker  docker build -t D ...