动态规划解决整数拆分问题

代码随想录链接:代码随想录

思路:

(1).确定dp数组的含义:

dp[i]:分拆数字i,可以得到的最大乘积为dp[i]

(2).确定递推公式:

dp[i]最大乘积是怎么得到:

其实可以从1遍历j,然后有两种渠道得到dp[i].

一个是**j * (i - j)**直接相乘。

一个是j * dp[i - j] ,相当于是拆分(i - j)

**因此递推公式为:**dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j))

注意这里在确定dp[i]时从1到i遍历j时更新dp[i]需要比较三个内容,分别是原始的dp[i],j*(i-j)以及j*dp[i-j],取这三个数的最大值更新dp[i]

(3).初始化dp数组:

严格从dp[i]的定义来说,dp[0] dp[1] 就不应该初始化,也就是没有意义的数值

只初始化dp[2] = 1,从dp[i]的定义来说,拆分数字2,得到的最大乘积是1,这个没有任何异议

(4).确定遍历顺序:

遍历i一定是从前向后遍历,先有dp[i - j]再有dp[i]

枚举j的时候,是从1开始的,从0开始的话,那么让拆分一个数拆个0,求最大乘积就没有意义

j的结束条件是 j < i - 1 ,其实 j < i 也是可以的

但是j的遍历只需要遍历到i/2(包含)就可以,后面就没有必要遍历

Java代码:

java 复制代码
class Solution {
    public int integerBreak(int n) {
        //dp[i] 为正整数 i 拆分后的结果的最大乘积
        int[] dp = new int[n+1];
        dp[2] = 1;
        for(int i = 3; i <= n; i++) {
            for(int j = 1; j <= i/2; j++) {
                // 这里的 j 其实最大值为 i-j,再大只不过是重复而已,
                //并且,在本题中,我们分析 dp[0], dp[1]都是无意义的,
                //j 最大到 i-j,就不会用到 dp[0]与dp[1]
                dp[i] = Math.max(dp[i], Math.max(j*(i-j), j*dp[i-j]));
                // j * (i - j) 是单纯的把整数 i 拆分为两个数 也就是 i,i-j ,再相乘
                //而j * dp[i - j]是将 i 拆分成两个以及两个以上的个数,再相乘。
            }
        }
        return dp[n];
    }
}
相关推荐
AI科技星10 分钟前
为什么变化的电磁场才产生引力场?—— 统一场论揭示的时空动力学本质
数据结构·人工智能·经验分享·算法·计算机视觉
TheLegendMe1 小时前
贪心+线程安全单例
算法·哈希算法
豐儀麟阁贵1 小时前
8.5在方法中抛出异常
java·开发语言·前端·算法
胖咕噜的稞达鸭2 小时前
算法入门:滑动窗口--->找到字符串中所有的字母异位词,串联所有的子串,最小覆盖子串
数据库·redis·算法
小青龙emmm2 小时前
2025级C语言第二次周测(国教专用)题解
c语言·开发语言·算法
WolfGang0073212 小时前
代码随想录算法训练营Day28 | 509.斐波那契数列、70.爬楼梯、746.使用最小花费爬楼梯
算法
Boop_wu3 小时前
[Java EE] 多线程进阶(JUC)(2)
java·jvm·算法
闻缺陷则喜何志丹3 小时前
【SOSDP模板 容斥原理 逆向思考】3757. 有效子序列的数量|分数未知
c++·算法·力扣·容斥原理·sosdp·逆向思考
CoovallyAIHub3 小时前
如何在手机上轻松识别多种鸟类?我们发现了更简单的秘密……
深度学习·算法·计算机视觉
第二只羽毛4 小时前
遵守robots协议的友好爬虫
大数据·爬虫·python·算法·网络爬虫