IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> Leetcode之1979.找出数组的最大公约数 -> 正文阅读

[数据结构与算法]Leetcode之1979.找出数组的最大公约数

题目

题目描述

给你一个整数数组 nums ,返回数组中最大数和最小数的 最大公约数 。
两个数的 最大公约数 是能够被两个数整除的最大正整数。

示例 1:

输入:nums = [2,5,6,9,10]
输出:2
解释:
nums 中最小的数是 2
nums 中最大的数是 10
2 和 10 的最大公约数是 2

示例 2:

输入:nums = [7,5,6,8,3]
输出:1
解释:
nums 中最小的数是 3
nums 中最大的数是 8
3 和 8 的最大公约数是 1

示例 3:

输入:nums = [3,3]
输出:3
解释:
nums 中最小的数是 3
nums 中最大的数是 3
3 和 3 的最大公约数是 3

提示:

  • 2 <= nums.length <= 1000
  • 1 <= nums[i] <= 1000

来源

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/find-greatest-common-divisor-of-array
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

题解

解法1

class Solution {
    /**
     * <p>思路:循环判断。</p>
     * <p>结果:成功</p>
     * <ul>
     *     <li>执行用时:0 ms, 在所有 Java 提交中击败了100.00% 的用户</li>
     *     <li>内存消耗:37.9 MB, 在所有 Java 提交中击败了96.92% 的用户</li>
     *     <li>通过测试用例:215 / 215</li>
     * </ul>
     *
     * @param nums 整数数组
     * @return 最大数和最小数的最大公约数
     */
    public int findGCD(int[] nums) {
        // 第一步,求得数组中的最大数和最小数
        int max = nums[0];
        int min = nums[0];
        for (int num : nums) {
            max = Math.max(max, num);
            min = Math.min(min, num);
        }

        // 第二步,判断最大公约数
        // 如果最大数对最小数取余,如果等于0那么最小数一定是最大公约数
        if (max % min == 0) {
            return min;
        } else {
            // 如果不等于,那么从最小数开始遍历,寻找最大公约数
            for (int i = min; i > 0; i--) {
                // 即最大数和最小数同时对i取余,为零者则是最大公约数
                if (max % i == 0 && min % i == 0) {
                    return i;
                }
            }
        }
        return 1;
    }
}

解法2

public class Solution {
    /**
     * <p>思路:辗转相除法。两个正整数a和b(a>b),它们的最大公约数等于a除以b的余数c和b之间的最大公约数。
     * 例如10和25,25除以10商2余5,那么10和25的最大公约数,等同于10和5的最大公约数。
     * 首先,计算出a除以b的余数c,把问题转化成求b和c的最大公约数;然后计算出b除以c的余数d,
     * 把问题转化成求c和d的最大公约数;再计算出c除以d的余数e,把问题转化成求d和e的最大公约数……以此类推,
     * 逐渐把两个较大整数之间的运算简化成两个较小整数之间的运算,直到两个数可以整除,或者其中一个数减小到1为止。</p>
     *
     * <p>结果:成功</p>
     * <ul>
     *     <li>执行用时:0 ms, 在所有 Java 提交中击败了100.00% 的用户</li>
     *     <li>内存消耗:37.9 MB, 在所有 Java 提交中击败了97.13% 的用户</li>
     *     <li>通过测试用例:215 / 215</li>
     * </ul>
     *
     * @param nums 整数数组
     * @return 最大数和最小数的最大公约数
     */
    public int findGCD(int[] nums) {
        // 第一步,求得数组中的最大数和最小数
        int max = nums[0];
        int min = nums[0];
        for (int num : nums) {
            max = Math.max(max, num);
            min = Math.min(min, num);
        }

        // 第二步,不断辗转相除
        return gcd(max, min);
    }

    /**
     * <p>求两个数的最大公约数,利用辗转相除法。</p>
     *
     * @param a 第一个整数
     * @param b 第二个整数
     * @return 最大公约数
     */
    private int gcd(int a, int b) {
        int max = Math.max(a, b);
        int min = Math.min(a, b);
        if (max % min == 0) {
            return min;
        }
        return gcd(max % min, min);
    }
}

解法3

public class Solution {
    /**
     * <p>思路:更相减损术:两个正整数a和b(a>b),它们的最大公约数等于a-b的差值c和较小数b的最大公约数。
     * 例如10和25,25减10的差是15,那么10和25的最大公约数,等同于10和15的最大公约数。由此,我们同样可以通过递归来简化问题。
     * 首先,计算出a和b的差值c(假设a>b),把问题转化成求b和c的最大公约数;然后计算出c和b的差值d(假设c>b),
     * 把问题转化成求b和d的最大公约数;再计算出b和d的差值e(假设b>d),把问题转化成求d和e的最大公约数……</p>
     * <p>结果:成功</p>
     * <ul>
     *     <li>执行用时:1 ms, 在所有 Java 提交中击败了67.14% 的用户</li>
     *     <li>内存消耗:37.6 MB, 在所有 Java 提交中击败了99.84% 的用户</li>
     *     <li>通过测试用例:215 / 215</li>
     * </ul>
     *
     * @param nums 整数数组
     * @return 最大数和最小数的最大公约数
     */
    public int findGCD(int[] nums) {
        // 第一步,求得数组中的最大数和最小数
        int max = nums[0];
        int min = nums[0];
        for (int num : nums) {
            max = Math.max(max, num);
            min = Math.min(min, num);
        }

        // 第二步,不断更相减损术
        return gcd(max, min);
    }

    /**
     * <p>求两个数的最大公约数,利用更相减损术。</p>
     *
     * @param a 第一个整数
     * @param b 第二个整数
     * @return 最大公约数
     */
    private int gcd(int a, int b) {
        if (a == b) {
            return a;
        }
        int max = Math.max(a, b);
        int min = Math.min(a, b);
        return gcd(max - min, min);
    }
}

解法4

public class Solution {
    /**
     * <p>思路:把辗转相除法和更相减损术的优势结合起来,在更相减损术的基础上使用移位运算。思想如下:
     * <ul>
     *     <li>当a和b均为偶数时,gcd(a,b) = 2×gcd(a/2, b/2) = 2×gcd(a>>1,b>>1)。</li>
     *     <li>当a为偶数,b为奇数时,gcd(a,b) = gcd(a/2,b) = gcd(a>>1,b)。</li>
     *     <li>当a为奇数,b为偶数时,gcd(a,b) = gcd(a,b/2) = gcd(a,b>>1)。</li>
     *     <li>当a和b均为奇数时,先利用更相减损术运算一次,gcd(a,b)=gcd(b,a-b),此时a-b必然是偶数,然后又可以继续进行移位运算。</li>
     * </ul>
     * </p>
     * <p>例如,计算10和25的最大公约数的步骤如下:</p>
     * <ul>
     *     <li>1. 整数10通过移位,可以转换成求5和25的最大公约数。</li>
     *     <li>2.利用更相减损术,计算出25-5=20,转换成求5和20的最大公约数。</li>
     *     <li>3. 整数20通过移位,可以转换成求5和10的最大公约数。</li>
     *     <li>4. 整数10通过移位,可以转换成求5和5的最大公约数。</li>
     *     <li>5. 利用更相减损术,因为两数相等,所以最大公约数是5。</li>
     * </ul>
     *
     * <p>结果:成功</p>
     * <ul>
     *     <li>执行用时:1 ms, 在所有 Java 提交中击败了67.14% 的用户</li>
     *     <li>内存消耗:38 MB, 在所有 Java 提交中击败了86.11% 的用户</li>
     *     <li>通过测试用例:215 / 215</li>
     * </ul>
     *
     * @param nums 整数数组
     * @return 最大数和最小数的最大公约数
     */
    public int findGCD(int[] nums) {
        // 第一步,求得数组中的最大数和最小数
        int max = nums[0];
        int min = nums[0];
        for (int num : nums) {
            max = Math.max(max, num);
            min = Math.min(min, num);
        }

        // 第二步
        return gcd(max, min);
    }

    /**
     * <p>求两个数的最大公约数,利用更相减损术。</p>
     *
     * @param a 第一个整数
     * @param b 第二个整数
     * @return 最大公约数
     */
    private int gcd(int a, int b) {
        if (a == b) {
            return a;
        }
        /*如果a和b都是偶数
            if(a%2==0&&b%2==0){
                return gcd(a/2, b/2)*2;
            }
         */
        if ((a & 1) == 0 && (b & 1) == 0) {
            return gcd(a >> 1, b >> 1) << 1;
        }
        /*如果a是偶数,b是奇数
            else if(a%2==0&&b%2!=0){
                return gcd(a/2, b);
            }
         */
        else if ((a & 1) == 0 && (b & 1) != 0) {
            return gcd(a >> 1, b);
        }
        /*如果a是奇数,b是偶数
            else if(a%2!=0&&b%2==0){
                return gcd(a, b/2);
            }
         */
        else if ((a & 1) != 0 && (b & 1) == 0) {
            return gcd(a, b >> 1);
        }
        /*如果a和b都是奇数
            else(a%2!=0&&a%2!=0){
                int max = Math.max(a, b);
                int min = Math.min(a, b);
                return gcd(max - min, min);
            }
         */
        else {
            int max = Math.max(a, b);
            int min = Math.min(a, b);
            return gcd(max - min, min);
        }
    }
}
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-09-12 13:24:22  更:2021-09-12 13:26:33 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/21 20:14:08-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码