求最大公约数的两种算法

1.辗转相除法

辗转相除法是求两个自然数的最大公约数的一种方法,也叫欧几里德算法。
例如,求gcd(319,377):
∵ 377÷319=1(余58)
∴gcd(377,319)=gcd(319,58);
∵ 319÷58=5(余29),
∴ gcd(319,58)=gcd(58,29);
∵ 58÷29=2(余0),
∴ gcd(58,29)= 29;
∴ gcd(319,377)=29

算法实现

//辗转相除法
int gcd(int m , int n){
    int r = 0 ;
    while(n){
        r = m%n;
        m = n;
        n = r;
    }
    return m;
}

2. 更相减损法:

更相减损术,是出自《九章算术》的一种求最大公约数的算法,它原本是为约分而设计的,但它适用于任何需要求最大公约数的场合。
《九章算术》是中国古代的数学专著,其中的“更相减损术”可以用来求两个数的最大公约数,即“可半者半之,不可半者,副置分母、子之数,以少减多,更相减损,求其等也。以等数约之。”
翻译成现代语言如下:
第一步:任意给定两个正整数;判断它们是否都是偶数。若是,则用2约简;若不是则执行第二步。
第二步:以较大的数减较小的数,接着把所得的差与较小的数比较,并以大数减小数。继续这个操作,直到所得的减数和差相等为止。
则第一步中约掉的若干个2与第二步中等数的乘积就是所求的最大公约数。
其中所说的“等数”,就是最大公约数。求“等数”的办法是“更相减损”法。所以更相减损法也叫等值算法。
例1、用更相减损术求98与63的最大公约数。
解:由于63不是偶数,把98和63以大数减小数,并辗转相减:
98-63=35
63-35=28
35-28=7
28-7=21
21-7=14
14-7=7
所以,98和63的最大公约数等于7。
这个过程可以简单的写为:
(98,63)=(35,63)=(35,28)=(7,28)=(7,21)=(7,14)=(7,7)=7.

例2、用更相减损术求260和104的最大公约数。
解:由于260和104均为偶数,首先用2约简得到130和52,再用2约简得到65和26。
此时65是奇数而26不是奇数,故把65和26辗转相减:
65-26=39
39-26=13
26-13=13
所以,260与104的最大公约数等于13乘以第一步中约掉的两个2,即13*2*2=52。
这个过程可以简单地写为:
(260,104)=(65,26)=(39,26)=(13,26)=(13,13)=13

算法实现

//更相减损术
int gcd(int m , int n){
    int mul=1;
    while(!(m&0x1) && !(n&0x1) ){
        m>>=1;
        n>>=1;
        mul<<=1;
    }
    int res=1;
    while((res =m-n)!=n){
        m = n>=res?n:res;
        n = n<=res?n:res;
    }
    return n*mul;

}

3. 辗转相除法与更相减损术的区别

(1)都是求最大公因数的方法,计算上辗转相除法以除法为主,更相减损术以减法为主,计算次数上辗转相除法计算次数相对较少,特别当两个数字大小区别较大时计算次数的区别较明显。
(2)从结果体现形式来看,辗转相除法体现结果是以相除余数为0则得到,而更相减损术则以减数与差相等而得到。

4.常用结论

在解有关最大公约数、最小公倍数的问题时,常用到以下结论:
  (1)如果两个数是互质数,那么它们的最大公约数是1,最小公倍数是这两个数的乘积。
  例如8和9,它们是互质数,所以(8,9)=1,[8,9]=72。
  (2)如果两个数中,较大数是较小数的倍数,那么较小数就是这两个数的最大公约数,较大数就是这两个数的最小公倍数。
  例如18与3,18÷3=6,所以(18,3)=3,[18,3]=18。
  (3)两个数分别除以它们的最大公约数,所得的商是互质数。
  例如8和14分别除以它们的最大公约数2,所得的商分别为4和7,那么4和7是互质数。
  (4)两个数的最大公约数与它们的最小公倍数的乘积等于这两个数的乘积。
  
  PS:由上述第一条结论可知:两个数的最小公倍数求法: m*n/gcd(gcd为最大公约数)

你可能感兴趣的:(数据结构与算法)