第1关:最小公倍数算法

任务描述

本关任务:编写一个能计算给定的所有正整数的最小公倍数的小程序。

相关知识

为了完成本关任务,你需要掌握:

  1. 如何求任意两个正整数的最大公约数;
  2. 如何求任意两个正整数的最小公倍数。

如何求任意两个正整数的最大公约数

最大公约数(GCD, Greatest Common Divisor),也称最大公因数、最大公因子,指两个或多个整数共有约数中最大的一个。

比如数12和数18的最大公约数是6,因为12的约数有1、2、3、4、6、12,而18的约数有1、2、3、6、9、18,通过比较,显然6是数12和数18的最大公约数。

通过上述过程,显然我们可以通过枚举这两个数的所有约数,考虑这两个数共有的约数,然后选择最大的就是这两个数的最大公约数。因为一个数的约数必然是不大于该数的,所以我们可以通过枚举不超过这两个数中的最大者的正整数,来达到上述效果,具体代码如下述所示:

  1. def gcd_1(x, y):
  2. ed = max(x, y)+1
  3. divisor = 1
  4. for i in range(2, ed):
  5. if x % i == 0 and y % i == 0:
  6. divisor = i
  7. return divisor

其实在古代就有能求解出最大公约数的算法了,《九章算术》是中国古代的数学专著,其中的“更相减损术”就可以用来求两个数的最大公约数,原文是:“可半者半之,不可半者,副置分母、子之数,以少减多,更相减损,求其等也。以等数约之。”大致所描述的算法步骤是:

  • 任意给定两个正整数;判断它们是否都是偶数。若是,则用2约简;若不是则执行第二步;
  • 以较大的数减较小的数,接着把所得的差与较小的数比较,并以大数减小数。继续这个操作,直到所得的减数和差相等为止;
  • 第一步中约掉的若干个2与第二步中最后得到的差(或减数)的乘积就是所求的最大公约数。

实际编程中,我们可以省略第一步,这样第二步最后得到的差(或减数)就是这两个数的最大公约数,其具体实现如下述代码所示:

  1. def gcd_2(x, y):
  2. while True:
  3. if x < y:
  4. x, y = y, x
  5. elif x == y:
  6. return x
  7. x -= y

尽管前面已经介绍了两种求最大公约数的方法,但实际生活中,我们更倾向于使用辗转相除法,来求解任意两个正整数的最大公约数,以求解3012的最大公约数为例,按gcd_2代码,其过程为:

30 - 12 = 18 -> 18 - 12 = 6 -> 12 - 6 = 6

最后因为减数和差相等,即6 - 6 = 0,故6就是3012的最大公约数,仔细观察上述过程,我们可以发现第一步和第二步实际上就是被减数30减了212,然后在第三步,用上次计算的余数6继续与12进行比较。显然,我们可以通过整数求余运算,直接一步求得3012的余数6,此时余数绝对是比除数小的,那么则将除数代替被除数的位置,余数代替除数的位置,然后重复上述过程,直至余数为0,那么此时的除数就是原来两个数的最大公约数了。上述过程用递归方式实现的话,代码是非常简短的,具体代码如下:

  1. def gcd(x, y):
  2. return x if y == 0 else gcd(y, x%y)

最后,推荐大家也去实现下求解任意两正整数的最大公约数的非递归版本。

如何求任意两个正整数的最小公倍数

几个数共有的倍数叫做这几个数的公倍数,其中除0以外最小的一个公倍数,叫做这几个数的最小公倍数(LCM,Least Common Multiple)。

37的最小公倍数是21,因为不存在一个比21还小的正整数,既是3的倍数,也是7的倍数。

显然对任意两个正整数aba*b必是他们的公倍数。假设gab的最大公约数,那么ab可以分别写成一个正整数与他们最大公约数的乘积的形式,即a = p * gb = q * g。那么显然c = p * q * g,是ab的一个公倍数,而且是最小公倍数。因为pq必定不共有大于1的公约数,所以若减小p、q、g这三个任意一个数的话,都不能使其乘积还是ab的倍数。

编程要求

根据提示,在右侧编辑器Begin-End区间补充代码,计算并输出给定的所有正整数的最小公倍数,参数x为整数列表。

本关涉及的代码文件src/step1/lcm_stu.py,请读者仔细阅读并完成空缺代码的填写。

测试说明

本关的测试文件是src/step1/main.py

  • 读者将src/step1/lcm_stu.py中的代码补充完毕,然后点击评测,平台自动编译运行src/step1/main.py,并以标准输入方式提供测评输入;
  • 平台获取程序的输出,然后将其与预期输出对比,如果一致则测试通过;否则测试失败。

平台会对你编写的代码进行测试:

测试输入: 1,2,3,4,5

预期输出: 60

测试输入: 15,25,20 预期输出: 300

解答

class Solution():
    def get_lcm(self, x):
        #请在此添加代码,实现求出给定的所有正整数的最小公倍数,并将其返回
        #********** Begin *********#
        def two_gcd(x, y):
            return x if y == 0 else two_gcd(y, x % y)
            # 递归版辗转相除法
        def two_lcm(x, y):
            return x * y / two_gcd(x, y)
            # 最小公倍数 乘 最大公约数 = 两数乘积
        for i in range(len(x) - 1):
            x[i + 1] = two_lcm(x[i], x[i + 1])
            # 先求最前面两个数的最小公倍数,再用该最小公倍数与后一个数求,依次进行
            result = x[i + 1]
        return int(result)
        #********** End **********#
        pass
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
最小公倍数(Least Common Multiple,简称LCM)是指两个或多个数中能够被所有这些数整除的最小正整数。在数据结构和算法中,求两个数的最小公倍数可以使用最大公约数(Greatest Common Divisor,简称GCD)的概念来解决。 根据引用的介绍,最小公倍数可以通过以下公式计算最小公倍数 = 两数的乘积 / 最大公约数。因此,我们首先需要计算出两个数的最大公约数。 引用给出了两种实现最大公约数的方法。第一种是非递归实现,其中使用了取余运算和循环来计算最大公约数。第二种是递归实现,其中使用了递归调用来计算最大公约数。 在非递归实现中,我们可以使用BigInt类来处理大整数。通过对输入的两个数进行比较,然后使用取余运算和循环来计算最大公约数。最后返回计算结果。 在递归实现中,我们通过判断其中一个数是否为0来终止递归。如果是0,则返回另一个数作为最大公约数;否则,继续递归调用,将第二个数和第一个数取余的结果作为参数传入。 接下来,我们可以根据最大公约数计算最小公倍数。根据引用的介绍,最小公倍数可以通过两个数的乘积除以最大公约数来计算。 因此,我们可以编写一个方法,接收两个数作为参数,先计算出最大公约数,然后将两个数的乘积除以最大公约数即可得到最小公倍数。 总结起来,计算两个数的最小公倍数的步骤如下: 1. 使用非递归或递归的方法计算出两个数的最大公约数。 2. 将两个数的乘积除以最大公约数,得到最小公倍数。 3. 返回最小公倍数作为结果。 希望这些信息对您有所帮助。如果您还有其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值