ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

辗转相除/更相减损/移位,求最大公约数

2021-05-20 22:02:59  阅读:165  来源: 互联网

标签:return gcd 更相 int 相除 else 减损 subtractV2 最大公约数


本文总结自:《漫画算法:小灰的算法之旅》魏梦舒

public class Main{

    public static void main(String[] args){
        int a = 35;
        int b = 21;
        System.out.println(getGreatestCommonDivisor(a, b));
        System.out.println(subtract(a, b));
        System.out.println(subtractV2(a, b));
    }

    private static int getGreatestCommonDivisor(int a, int b) {
        int result;
        if (a > b) {
            result = divide(a, b);
        } else {
            result = divide(b, a);
        }
        return result;
    }

    private static int divide(int a, int b) {
        if (a % b == 0) {
            return b;
        } else {
            //辗转相除法:两个正整数a和b(a>b),它们的最大公约数等于 a除以b的余数c 和 b 之间的最大公约数
            //以此类推,逐渐把两个较大整数之间的运算简化成两个较小整数之间的运算,直到两个数可以整除,或者其中一个数减小到1为止
            //缺点:当两个整数较大时,a%b性能较低
            return divide(b, a % b);
        }
    }

    private static int subtract(int a, int b) {
        if (a == b) {
            return a;
        } else if (a > b) {
            //更相减损法:两个正整数a和b(a>b),它们的最大公约数等于 a-b的差值c 和 较小数b 的最大公约数
            //以此类推,逐渐把两个较大整数之间的运算简化成两个较小整数之间的运算,直到两个数可以相等为止,最大公约数就是最终相等的两个数
            //缺点:不稳定的算法,当两数相差悬殊时,比如计算10000和1,就要递归9999次
            return subtract(a - b, b);
        } else {
            return subtract(b - a, a);
        }
    }

    private static int subtractV2(int a, int b) {
        if (a == b) {
            return a;
        }
        //保证a > b
        if (a < b) {
            return subtractV2(b, a);
        }
        //位与运算判断奇偶
        if ((a & 1) == 0 && (b & 1) == 0) {
            //当a和b均为偶数,gcd(a,b) = 2*gcd(a/2, b/2) = 2*gcd(a>>1, b>>1)
            return subtractV2(a >> 1, b >> 1) << 1;
        } else if ((a & 1) == 0) {
            //当a为偶数,b为奇数,gcd(a,b) = gcd(a/2, b) = gcd(a>>1, b)
            return subtractV2(a >> 1, b);
        } else if ((b & 1) == 0) {
            //当a为奇数,b为偶数,gcd(a,b) = gcd(a, b/2) = gcd(a, b>>1)
            return subtractV2(b >> 1, a);
        } else {
            //当a和b均为奇数,利用更相减损法运算一次,gcd(a,b) = gcb(a - b, b),此时a-b必然是偶数,又可以继续进行移位运算。
            return subtractV2(a - b, b);
        }
    }

}

标签:return,gcd,更相,int,相除,else,减损,subtractV2,最大公约数
来源: https://blog.csdn.net/weixin_42446602/article/details/117092447

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有