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

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

思路:

(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];
    }
}
相关推荐
ZhengEnCi2 小时前
S10-蓝桥杯 17822 乐乐的积木塔
算法
贾斯汀玛尔斯2 小时前
每天学一个算法--拓扑排序(Topological Sort)
算法·深度优先
大龄程序员狗哥2 小时前
第25篇:Q-Learning算法解析——强化学习中的经典“价值”学习(原理解析)
人工智能·学习·算法
exp_add32 小时前
质数相关知识
算法
小辉同志3 小时前
215. 数组中的第K个最大元素
数据结构·算法·leetcode··快速选择
小O的算法实验室3 小时前
2025年IEEE TITS,基于矩阵的进化计算+面向无线传感器网络数据收集无人机路径规划,深度解析+性能实测
算法·论文复现·智能算法·智能算法改进
OidEncoder4 小时前
编码器分辨率与机械精度的关系
人工智能·算法·机器人·自动化
memcpy04 小时前
LeetCode 2615. 等值距离和【相同元素分组+前缀和;考虑距离和的增量】中等
算法·leetcode·职场和发展
炽烈小老头4 小时前
【 每天学习一点算法 2026/04/22】四数相加 II
学习·算法
alphaTao4 小时前
LeetCode 每日一题 2026/4/20-2026/4/26
算法·leetcode·职场和发展