欧几里德算法与扩展欧几里德算法

欧几里得算法就是我们常说的辗转相除法,辗转相除法可以用来求最大公约数,知道最大公约数还可以求最小公倍数。gcd在好像也有库函数__gcd

int Gcd(int a, int b)
{
    while(b != 0)
    {
      int r = b;
      b = a % b;
      a = r;
    }
    return a;
}
非递归方法
int gcd(int a,int b)
{return b ? gcd(b,a%b) : a;}
递归方法

扩展欧几里得算法是在解决一个什么问题呢,解方程,解二元一次方程的通解。

扩展欧几里得算法是欧几里得算法(又叫辗转相除法)的扩展。除了计算a、b两个整数的最大公约数,此算法还能找到整数x、y(其中一个很可能是负数)。通常谈到最大公因子时, 我们都会提到一个非常基本的事实: 给予二整数 a 与 b, 必存在有整数 x 与 y 使得ax + by = gcd(a,b)。有两个数a,b,对它们进行辗转相除法,可得它们的最大公约数——这是众所周知的。然后,收集辗转相除法中产生的式子,倒回去,可以得到ax+by=gcd(a,b)的整数解。by百度百科

int exgcd(int a,int b,int &x,int &y)
{
    if(b==0)
    {
        x=1;
        y=0;
        return a;
    }
    int r=exgcd(b,a%b,x,y);
    int t=x;
    x=y;
    y=t-a/b*y;
    return r;
}
非递归方法
int exgcd(int m,int n,int &x,int &y)
{
    int x1,y1,x0,y0;
    x0=1; y0=0;
    x1=0; y1=1;
    x=0; y=1;
    int r=m%n;
    int q=(m-r)/n;
    while(r)
    {
        x=x0-q*x1; y=y0-q*y1;
        x0=x1; y0=y1;
        x1=x; y1=y;
        m=n; n=r; r=m%n;
        q=(m-r)/n;
    }
    return n;
}
递归方法

扩展欧几里德算法的应用主要有以下三方面:

(1)求解不定方程;

(2)求解模线性方程(线性同余方程);

(3)求解模的逆元;

模逆元的代码,当时用来加AA小姐接的QQ

#include <stdio.h>
__int64 la(__int64 a,__int64 b,__int64 &x,__int64 &y){
    __int64 d;
    if(!b){
        x=1;y=0;return a;
    }
    d=la(b,a%b,y,x);
    y-=a/b*x;
    return d;
    
}
__int64  Inv(__int64 a,__int64 n){
    __int64 d,x,y;
    d=la(a,n,x,y);
    if(d==1)
    return (x%n+n)%n;
    else
    return -1;
}
int main()
{   __int64 a,n;
    scanf("%I64d%I64d",&a,&n);
    printf("%I64d",Inv(a,n));
    
    return 0;
}

 

posted @ 2017-05-13 13:11  暴力都不会的蒟蒻  阅读(406)  评论(0编辑  收藏  举报