最大切分乘积问题
最大切分乘积问题
注意
给定一个正整数 $n$ ,将其切分为至少两个正整数的和,求切分后所有整数的乘积最大是多少。
假设我们将 切分为 个整数因子,其中第 个因子记为 ,即
本题目标是求得所有整数因子的最大乘积,即
我们需要思考的是:切分数量 应该多大,每个 应该是多少?
贪心策略确定
根据经验,两个整数的乘积往往比它们的加和更大。假设从 中分出一个因子 ,则它们的乘积为 。我们将该乘积与 作比较:
如下图所示,当 时,切分出一个 后乘积会变大,这说明大于等于 的整数都应该被切分。
贪心策略一:如果切分方案中包含 的因子,那么它就应该被继续切分。最终的切分方案只应出现 、、 这三种因子。
接下来思考哪个因子是最优的。在 、、 这三个因子中,显然 是最差的,因为 恒成立,即切分出 反而会导致乘积减小。
如下图所示,当 时,有 。这意味着切分出 比切分出 更优。
贪心策略二:在切分方案中,最多只应存在两个 。因为三个 总是可以被替换为两个 ,从而获得更大乘积。
总结以上,可推出以下贪心策略。
- 输入整数 ,从其不断地切分出因子 ,直至余数为 、、 。
- 当余数为 时,代表 是 的倍数,因此不做任何处理。
- 当余数为 时,不继续划分,保留之。
- 当余数为 时,由于 ,因此应将最后一个 替换为 。
代码实现
如下图所示,我们无须通过循环来切分整数,而可以利用向下整除运算得到 的个数 ,用取模运算得到余数 ,此时有:
请注意,对于 的边界情况,必须拆分出一个 ,乘积为 。
[file]{max_product_cutting}-[class]{}-[func]{max_product_cutting}
时间复杂度取决于编程语言的幂运算的实现方法。以 Python 为例,常用的幂计算函数有三种。
- 运算符
**
和函数pow()
的时间复杂度均为 。 - 函数
math.pow()
内部调用 C 语言库的pow()
函数,其执行浮点取幂,时间复杂度为 。
变量 和 使用常数大小的额外空间,因此空间复杂度为 。
正确性证明
使用反证法,只分析 的情况。
- 所有因子 :假设最优切分方案中存在 的因子 ,那么一定可以将其继续划分为 ,从而获得更大的乘积。这与假设矛盾。
- 切分方案不包含 :假设最优切分方案中存在一个因子 ,那么它一定可以合并入另外一个因子中,以获取更大乘积。这与假设矛盾。
- 切分方案最多包含两个 :假设最优切分方案中包含三个 ,那么一定可以替换为两个 ,乘积更大。这与假设矛盾。