给定一个正整数 n,将其拆分为至少两个正整数的和,并使这些整数的乘积最大化。 返回你可以获得的最大乘积。
示例 1:
输入: 2
输出: 1
解释: 2 = 1 + 1, 1 × 1 = 1。
示例 2:
输入: 10
输出: 36
解释: 10 = 3 + 3 + 4, 3 × 3 × 4 = 36。
说明: 你可以假设 n 不小于 2 且不大于 58。
将正整数分割成几份,这个是未知的,所以很难用循环去解决,因为不知道要做几重循环,这种情况通常要使用递归,也就是暴力解法:回溯遍历将一个数做分割的所有可能性,时间复杂度为 O(2^n),但是容易产生重复子问题,如下图所示:
所以在此基础上可以用自顶向下的记忆化搜索或者自底向上的动态规划
res = max(res, i * backtrack(n-i))
;此处要注意一个陷阱:backtrack()这个函数的定义是至少要将n-i分割成两部分,可是当我们具体分割n时,就将n分割成 i和n-i 也是可以的,因此在取max时还需添加一个i*(n-i):res = max(res, i*(n-i), i*backtrack(n-i))
。也就是说对n分割成 i 和 n-i,(1)是不继续分割n-i了,直接看i*(n-i)哪一个最大;(2)是继续分割n-i,看能否找到一个分割后的结果与i的乘积最大。
但是上面这样写同样包含重叠子问题,重叠的部分就在于backtrack(n-i)部分,所以我们设置一个变量memo来具体的记录出现的结果,初始化为memo = [-1 for i in range(n+1)]
:
memo[n] != -1
,也就表明memo[n]之前计算过,直接返回memo[n]即可;memo[n] = res
),表示对这一次计算的结果进行了记录,再把res返回即可。——这就是记忆化搜索。保险起见,在主函数里运用assert,保证传进来的数字是大于等于2的 assert (n >=2),因为至少要分割成两部分,2是最小的可以分割成两部分的数字。
class Solution:
def integerBreak(self, n: int) -> int:
assert(n >= 2)
self.memo = [-1 for i in range(n + 1)]
return self.backtrack(n)
def backtrack(self, n):
if n == 1:
return 1
if self.memo[n] != -1:
return self.memo[n]
res = float('-inf')
for i in range(1, n):
res = max(res, i*(n-i), i*self.backtrack(n-i))
self.memo[n] = res
return res
在主函数中进行assert之后依然要创建一个记忆的数组memo,memo = [-1 for i in range(n+1)]
(表示memo有n+1个元素,每个元素的初始值为-1),memo[i]表示将数字i分割(至少分割成两部分)后得到的最大乘积。
使用动态规划是自底向上解决问题:
for i in range(2, n+1)
,具体求解的方式依然是再进行一次循环,这次循环是尝试将i这个数字进行分割:for j in range(1, i)
,也就是说每次都尝试将 i 这个数字分割成 j + (i - j)这样的两部分——对于这样的分割,背后的乘积依然是:(1)就是 j * (i-j),(2)就是将 i-j 继续分割得到 j * memo[i-j],因为 i-j 一定小于i,所以此时 memo[i-j] 一定已经被计算出来了,此处就可以被使用。memo[i] = max(memo[i], j * (i-j), j * memo[i-j])
,最终只需要返回memo[n]即可。class Solution:
def integerBreak(self, n: int) -> int:
assert(n >= 2)
memo = [-1 for i in range(n + 1)]
if n == 1:
return 1
for i in range(2, n + 1):
for j in range(1, i):
memo[i] = max(memo[i], j*(i-j), j*memo[i-j])
return memo[n]
记忆化搜索和动态规划的for循环起始值的差别:
# 记忆化搜索
for i in range(1, n):
res = max(res, i*(n-i), i*self.backtrack(n-i))
记忆化搜索 i 的起始值从1开始,是因为要将 n 分割成 (n-1)+1、(n-2)+2……这些情况,然后再对n-1、n-2……这部分进行分割,若 i 从2开始, 就会漏了 (n-1)+1 这种情况,结果会出错。
# 动态规划
for i in range(2, n + 1):
for j in range(1, i):
memo[i] = max(memo[i], j*(i-j), j*memo[i-j])
动态规划的外层循环 i 的起始值从2开始,是因为此处求解的是memo[n],而memo[1]的值已经知晓是为1,所以自然要从2开始;内循环是具体的分割部分,还是从1开始。