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

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

思路:

(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];
    }
}
相关推荐
那个村的李富贵14 小时前
CANN加速下的AIGC“即时翻译”:AI语音克隆与实时变声实战
人工智能·算法·aigc·cann
power 雀儿14 小时前
Scaled Dot-Product Attention 分数计算 C++
算法
琹箐14 小时前
最大堆和最小堆 实现思路
java·开发语言·算法
renhongxia115 小时前
如何基于知识图谱进行故障原因、事故原因推理,需要用到哪些算法
人工智能·深度学习·算法·机器学习·自然语言处理·transformer·知识图谱
坚持就完事了15 小时前
数据结构之树(Java实现)
java·算法
算法备案代理15 小时前
大模型备案与算法备案,企业该如何选择?
人工智能·算法·大模型·算法备案
赛姐在努力.15 小时前
【拓扑排序】-- 算法原理讲解,及实现拓扑排序,附赠热门例题
java·算法·图论
野犬寒鸦16 小时前
从零起步学习并发编程 || 第六章:ReentrantLock与synchronized 的辨析及运用
java·服务器·数据库·后端·学习·算法
霖霖总总16 小时前
[小技巧66]当自增主键耗尽:MySQL 主键溢出问题深度解析与雪花算法替代方案
mysql·算法
rainbow688917 小时前
深入解析C++STL:map与set底层奥秘
java·数据结构·算法