逆元与拓展欧几里得算法

· · 个人记录

一:乘法逆元定义:

若整数 b,p 互质,并且对于任意的整数 a,如果满足 b|a,则存在一个整数 x,使得 a/b≡a×x(mod p),则称 x 为 b 的模 p 乘法逆元,记为 b−1(mod p)。 b 存在乘法逆元的充要条件是 b 与模数 p 互质。当模数 p 为质数时,b^(p−2)即为 b 的乘法逆元。

二:快速幂求a%p逆元(p为质数):

推导:

因为a/b≡a*x(mod p)

对等式两边同时乘b得到:a≡abx(mod p)

同时约去a,得到:1≡b*x(mod p)

所以 1(mod p)=b*x

由费马小定理得到:b^(n-1)≡1(mod p)

同时提出一个b,得到:b*b^(n-2)≡1(mod p)

对比斜体式子得知,x=b^(n-2)

知识补充:

可能有同学好奇费马小定理是啥,这里来补充:

如果我们定义一个p为素数,且定义a和b,a%p≠0,则a^p-1≡1(mod p)。

接下来数学不好者慎入,我们要来证明他了:

首先创造一个集合x,包含所有小于p得数,显然p与集合里的所有数互质。

我们在定义一个集合X,用公式表示是a*x{1,......,p-1}%p=X{a%p,2a%p,......,(p-1)a%p},

将两个集合里的数分别相乘:(p-1)! % p=a^(p-1)*(p-1)! % p

把所有(p-1)!去掉,得到这个式子:

1(mod p)≡a^(p-1)(mod p),可以推出式子

a^(p-1)≡1(mod p)。

代码:

有了上述储备,我们可以很轻松的写出代码了。

#include <iostream>
using namespace std;

int quick(long long a,long long b,long long c)
{
    long long res=1;
    while(b)
    {
        if(b&1) res=res*a%c;  //此二进制位为1
        b>>=1;                //b右移一位
        a=a*a%c;             //a扩大一次方
    }

    return res;
}

int main()
{
    int n;
    cin>>n;
    while(n--)
    {
        long long  a,b;
        cin>>a>>b;
        if(a%b==0) cout<<"impossible"<<endl;
        else cout<<quick(a,b-2,b)<<endl;
    }
}

三:扩展欧几里得算法求a%p逆元(p不是质数)

裴蜀定理: 若a,b是整数,且gcd(a,b)=d,那么对于任意的整数x,y,ax+by都一定是d的倍数,特别地,一定存在整数x,y,使ax+by=d成立。 它的一个重要推论是:a,b互质的充分必要条件是存在整数x,y使ax+by=1. 扩展欧几里得算法:

适用:求解方程 ax+by=gcd(a,b) 的解

易得:b=0时:gcd(a,b)=a

所以x=1,y=0.

b!=0时:由欧几里得算法得到:gcd(a,b)=gcd(b,a%b)

bx′+(a%b)y′=gcd(b,a%b)

bx′+(a−⌊a/b⌋∗b)y′=gcd(b,a%b)

ay′+b(x′−⌊a/b⌋∗y′)=gcd(b,a%b)=gcd(a,b)

得出:x=y′,y=x′−⌊a/b⌋∗y′ 因此可以采用递归,使得某一层中,b=0,即可求出数值。 扩展欧几里得算法对逆元的帮助:

a*x≡1(mod p)

ax=-y*p+1

ax+py=1

代码:

#include <iostream>
using namespace std;

int exgcd(long long a,long long b,int &x,int &y)
{
    if(!b)
    {
        x=1,y=0;
        return a;//返回gcd=a
    }
    int d=exgcd(b,a%b,y,x);
    y-=a/b*x;
    return d;//返回gcd=d
}
int main()
{
    int n;
    cin>>n;
    while(n--)
    {
        long long a,b;
        int x,y;
        cin>>a>>b;
        int d=exgcd(a,b,x,y)==1;
        if(d==1) cout<<(x+b)%b<<endl;//保证x>0
        else cout<<"impossible"<<endl;

    }
}