算法学习笔记(37)——扩展欧几里得算法

扩展欧几里得算法

欧几里得算法/辗转相除法(Euclidean algorithm)

辗转相除法,又称欧几里得算法(英语:Euclidean algorithm),是求最大公约数的算法。

两个整数的最大公约数能够同时整除它们的最大的正整数。辗转相除法基于如下原理:两个整数的最大公约数等于其中较小的数和两数相除余数的最大公约数

例如,252和105的最大公约数是21
252=21×12;105=21×5
252105=21×(125)=147,所以147和105的最大公约数也是21。
在这过程中,较大的数缩小了,所以继续进行同样的计算可以不断缩小这两个数直至余为零。
注意:0与任何数的最大公约数即为该数
这时,所剩下的还没有变成零的数就是两数的最大公约数。

裴蜀定理(Bézout 定理)

两数的最大公约数可以用两数的整数倍相加来表示,如21=5×105+(2)×252

对任意整数 a,b,存在一对整数 x,y,满足 ax+by=gcd(a,b)

证明:

在欧几里得算法的最后一步,即 b=0 时,显然有一对整数 x=1,y=0,使得 a1+00=gcd(a,0)

b>0,则 gcd(a,b)=gcd(b,amodb);
假设存在一对整数 x,y,满足 bx+(amodb)y=gcd(b,amodb);
由于 bx+(amodb)y=bx+(aba/b)y=ay+b(xa/b)y;
则令 x=y,y=(xa/b)y;
即得到ax+by=gcd(a,b);
对欧几里得算法的递归过程应用数学归纳法,可知裴蜀定理成立。

证毕

扩展欧几里得算法(Extended Euclidean algorithm)

扩展欧几里得算法(英语:Extended Euclidean algorithm)是欧几里得算法(又叫辗转相除法)的扩展。已知整数a、b,扩展欧几里得算法可以在求得a、b的最大公约数的同时,能找到整数x、y(其中一个很可能是负数),使它们满足贝祖等式

ax+by=gcd(a,b)

假设 d=gcd(a,b),当求出一组x与y的值之后,即可求出所有的x与y的值:

a(xbdk)+b(y+adk)=ax+byabdk+badk=ax+by=gcd(a,b)

故所有的x与y的值可以用已经求出的一组解 (x0,y0) 表示为:

{x=x0bdky=y0+adk

// 求a与b的最大公约数,并求出满足裴蜀定理 ax+by=gcd(a,b) 的一组x与y的值
int exgcd(int a, int b, int &x, int &y)
{
    /* 
     * 确定边界条件,如果b==0,则gcd(a,b)=a,则a*1+b*0=a一定满足裴蜀定理
     * 这种边界情况在代码中是作为递归的最底层出现的
     * 这意味着当这一层解出来的时候,我们需要将其返回到上一层的解,直到返回至最开始的一层。
     */
    if (b == 0) { x = 1, y = 0; return a; }
    
    /*
     * 如果 b>0, gcd(a, b) = gcd(b, a%b)
     * b*x + (a % b)y = b*x + (a - a / b * b)*y = gcd(a, b)
     * a*y + b(x - a / b * y) = gcd(a, b)
     * x' = y, y' = x - a/b*y
     */
    int d = exgcd(b, a % b, x, y);
    int z = x, y = x, x = z - a / b * y;
    return d;
}

精简版:

int exgcd(int a, int b, int &x, int &y)
{
    if (b == 0) { x = 1, y = 0; return a; }
    // 在计算 b 与 a%b 的最大公约数时交换 x 与 y 的值,方便后续计算
    int d = exgcd(b, a % b, y, x);
    y -= a / b * x;
    return d;
}

题目链接:AcWing 877. 扩展欧几里得算法

#include <iostream>

using namespace std;

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, y, x);
    y -= a / b * x;
    return d;
}

int main()
{
    int n;
    scanf("%d", &n);
    while (n -- ) {
        int a, b, x, y;
        scanf("%d%d", &a, &b);
        exgcd(a, b, x, y);
        printf("%d %d\n", x, y);
    }
    return 0;
}

求解线性同余方程

给定整数 a,b,m,求一个整数 x 满足 axb(modm) ,或者给出无解。因为未知数的指数为 1 ,所以我们称之为一次同余方程,也称线性同余方程。

axb(modm) 等价于 axbm 的倍数,不妨设为 y 倍。则 ax=m(y)+b ,于是,该方程可以改写为 ax+my=b

则该问题转化为,给定 a,m,b ,求出一组x与y的值,使得 ax+my=b 成立。
而裴蜀定理则为,非定 a,m,gcd(a,m),求出一组x与y的值,使得 ax+my=gcd(a,m) 成立

根据裴蜀定理及其证明过程,线性同余方程有解当且仅当 gcd(a,m)|b

所以求解过程如下,记 d=gcd(a,m)

  1. 求出 ax+my=d 的一组特解 x0,y0(扩展欧几里得算法)
  2. 由于 d|b , 则令 x0,y0 同时乘以 b/d,就得到了 ax+my=b 的一组特解 bdx0,bdy0

事实上,ax+my=b 的通解可以表示为:

{x=bdx0+kmdy=bdy0+kad

其中 $ k \in \mathbb{Z}d = gcd(a, m)x_0, y_0$ 是ax+my=d 的一组特解。

题目链接:AcWing 878. 线性同余方程

#include <iostream>

using namespace std;

typedef long long LL;

// 扩展欧几里得算法
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, y, x);
    y -= a / b * x;
    return d;
}

int main()
{
    int n;
    scanf("%d", &n);
    
    while (n -- ) {
        int a, b, m;
        scanf("%d%d%d", &a, &b, &m);
        int x, y;
        int d = exgcd(a, m, x, y);
        // 线性同余方程有解的充要条件:b是d的倍数
        if (b % d == 0) printf("%d\n", (LL)b / d * x % m);
        else puts("impossible");
    }
    
    return 0;
}
posted @   S!no  阅读(101)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
· .NET周刊【3月第1期 2025-03-02】
点击右上角即可分享
微信分享提示