leetCode 343.整数拆分 动态规划

news2024/11/18 23:36:58

给定一个正整数 n ,将其拆分为 k 个 正整数 的和( k >= 2 ),并使这些整数的乘积最大化。

返回 你可以获得的最大乘积 。

示例 1:

输入: n = 2
输出: 1
解释: 2 = 1 + 1, 1 × 1 = 1。

示例 2:

输入: n = 10
输出: 36
解释: 10 = 3 + 3 + 4, 3 × 3 × 4 = 36。

>>动规五部曲

1.确定dp数组(dp table)以及下标的定义

  • dp[i]: 分拆数字i,可以得到的最大乘积为 dp[i]
  • dp[i] 的定义将贯彻整个解题过程

2.确定递推公式

 【思考🤔】如何得到上图这种拆分的结果呢?

详细分析:
dp[3] = max(dp[3],max((3-1)*1,dp[(3-1)] * 1)) = 2

dp[4] = max(dp[4],max((4-1)*1,dp[(4-1)] * 1)) = 3
dp[4] = max(dp[4],max((4-2)*2,dp[(4-2)] * 2)) = 4

dp[5] = max(dp[5],max((5-1)*1,dp[(5-1)] * 1)) = 4
dp[5] = max(dp[5],max((5-2)*2,dp[(5-2)] * 2)) = 6
dp[5] = max(dp[5],max((5-3)*3,dp[(5-3)] * 3)) = 6

dp[6] = max(dp[6],max((6-1)*1,dp[(6-1)] * 1)) = 6
dp[6] = max(dp[6],max((6-2)*2,dp[(6-2)] * 2)) = 8
dp[6] = max(dp[6],max((6-3)*3,dp[(6-3)] * 3)) = 9
dp[6] = max(dp[6],max((6-4)*4,dp[(6-4)] * 4)) = 9

dp[7] = max(dp[7],max((7-1)*1,dp[(7-1)] * 1)) = 9
dp[7] = max(dp[7],max((7-2)*2,dp[(7-2)] * 2)) = 12
dp[7] = max(dp[7],max((7-3)*3,dp[(7-3)] * 3)) = 12
dp[7] = max(dp[7],max((7-4)*4,dp[(7-4)] * 4)) = 12
dp[7] = max(dp[7],max((7-5)*5,dp[(7-5)] * 5)) = 12

dp[8] = max(dp[8],max((8-1)*1,dp[(8-1)] * 1)) = 12
dp[8] = max(dp[8],max((8-2)*2,dp[(8-2)] * 2)) = 18
dp[8] = max(dp[8],max((8-3)*3,dp[(8-3)] * 3)) = 18
dp[8] = max(dp[8],max((8-4)*4,dp[(8-4)] * 4)) = 18
dp[8] = max(dp[8],max((8-5)*5,dp[(8-5)] * 5)) = 18
dp[8] = max(dp[8],max((8-6)*6,dp[(8-6)] * 6)) = 18

dp[9] = max(dp[9],max((9-1)*1,dp[(9-1)] * 1)) = 18
dp[9] = max(dp[9],max((9-2)*2,dp[(9-2)] * 2)) = 24
dp[9] = max(dp[9],max((9-3)*3,dp[(9-3)] * 3)) = 27
dp[9] = max(dp[9],max((9-4)*4,dp[(9-4)] * 4)) = 27
dp[9] = max(dp[9],max((9-5)*5,dp[(9-5)] * 5)) = 27
dp[9] = max(dp[9],max((9-6)*6,dp[(9-6)] * 6)) = 27
dp[9] = max(dp[9],max((9-7)*7,dp[(9-7)] * 7)) = 27

dp[10] = max(dp[10],max((10-1)*1,dp[(10-1)] * 1)) = 27
dp[10] = max(dp[10],max((10-2)*2,dp[(10-2)] * 2)) = 27
dp[10] = max(dp[10],max((10-3)*3,dp[(10-3)] * 3)) = 36
dp[10] = max(dp[10],max((10-4)*4,dp[(10-4)] * 4)) = 36
dp[10] = max(dp[10],max((10-5)*5,dp[(10-5)] * 5)) = 36
dp[10] = max(dp[10],max((10-6)*6,dp[(10-6)] * 6)) = 36
dp[10] = max(dp[10],max((10-7)*7,dp[(10-7)] * 7)) = 36
dp[10] = max(dp[10],max((10-8)*8,dp[(10-8)] * 8)) = 36

动态递推公式: dp[i] = max(dp[i],max((i - j) * j,dp[i - j] * j));

3.dp数组的初始化

dp[2] = 1,从dp[i]的定义来说,拆分数字2,得到的最大乘积是1

4.确定遍历顺序

遍历顺序为: 

for (int i = 3; i <= n ; i++) {
    for (int j = 1; j < i - 1; j++) {
        dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j));
    }
}

注意:为什么 j=1;j < i - 1,因为dp[0] = 0,dp[1] = 0,所以无需拆分出 数字 0 或者 1,这对于求最大乘积是没有意义了!!!

j 的结束条件是 j < i - 1 ,其实 j < i 也是可以的,不过可以节省一步,例如让j = i - 1,的话,其实在 j = 1的时候,这一步就已经拆出来了,重复计算,所以 j < i - 1

 >>进一步优化

for (int i = 3; i <= n ; i++) {
    for (int j = 1; j <= i / 2; j++) {
        dp[i] = max(dp[i], max((i - j) * j, dp[i - j] * j));
    }
}

来自代码随想录的解释:代码随想录 (programmercarl.com)

因为拆分一个数n 使乘积最大,那么一定是拆分成 m 个近似相同的子数相乘才是最大的

例如 6 拆成 3 x 3,10 拆成 3 x 3 x 4。 100 的话 也是拆成 m 个近似数组的子数 相乘才是最大的。只不过我们不知道 m 究竟是多少而已,但是可以明确的是 m 一定 大于等于2,既然 m 大于等于 2 ,也就是 最差也应该是拆成两个相同的 可能是最大值。

那么 j 遍历,只需要遍历到 n / 2就可以,后面就没有必要遍历了,一定不是最大值

5.举例推导dp数组

 

// leetCode 343.整数拆分
class Solution {
public:
    // 动态规划 
    int integerBreak(int n) {
        vector<int> dp(n+1);
        dp[2] = 1;
        for(int i = 3;i <= n;i++) {
            for(int j = 1;j < i-1;j++) {
                dp[i] = max(dp[i],max((i-j)*j,dp[i-j]*j));
            }
        }
        return dp[n];
    }

    // 动态规划 + 优化
    int integerBreak(int n) {
        vector<int> dp(n+1);
        dp[2] = 1;
        for(int i = 3;i <= n;i++) {
            for(int j = 1;j <= i / 2;j++) {
                dp[i] = max(dp[i],max((i-j)*j,dp[i-j]*j));
            }
        }
        return dp[n];
    }
};


  • 时间复杂度:O(n^2)
  • 空间复杂度:O(n)

有贪心算法的解法,后续填坑~🕳

来自代码随想录课堂截图:

参考和推荐文章:

代码随想录 (programmercarl.com)

动态规划,本题关键在于理解递推公式!| LeetCode:343. 整数拆分_哔哩哔哩_bilibili

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1046833.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python实现IP的自动切换

一、安装所需库 在开始之前&#xff0c;我们首先需要确保已经安装了以下库&#xff1a; - requests&#xff1a;用于发送HTTP请求和获取网页内容。 - winreg&#xff1a;用于在Windows下访问和编辑注册表信息。 可以使用pip命令进行安装&#xff0c;例如&#xff1a; pip i…

PayPal面经

文章目录 初战AI Infra团队广泛收集信息&#xff0c;增加对面试相关团队的了解Paypal的AI infra Engineer 极客时间演讲视频&#xff1a;AI在金融应用HR面试首面 zhang chao首先让我介绍自己和项目基础知识出题 lettcode 1and0s 二面 luwen没有让我重复介绍自己那好&#xff0c…

linux操作系统--常用命令篇(网络安全、运维和测试人员必备技能)

前言&#xff1a;linux 命令是对 Linux 系统进行管理的命令。对于 Linux 系统来说&#xff0c;无论是中央处理器、内存、磁盘驱动器、键盘、鼠标&#xff0c;还是用户等都是文件&#xff0c; Linux 系统管理的命令是它正常运行的核心&#xff0c;与之前的 DOS 命令类似。linux …

设计模式 - 策略模式

目录 一. 前言 二. 实现 一. 前言 策略模式 (Strategy Pattern) 是指对一系列的算法定义&#xff0c;并将每一个算法封装起来&#xff0c;而且使它们还可以相互替换。此模式让算法的变化独立于使用算法的客户。 与状态模式的比较 状态模式的类图和策略模式类似&#xff0c;并…

sentinel 以及 sentinel-golang 让你的服务坚如磐石

首先要说 Sentinel&#xff0c;这是阿里巴巴内部使用多年并演化出来的流控软件&#xff0c;经受住了多年的双十一考验&#xff0c;最早是服务于Java语言的&#xff0c;在2020年推出了 Sentinel-golang 版本。 官方文档&#xff1a;https://sentinelguard.io/zh-cn/docs/introd…

GreatSQL一个关于主从复制的限制描述与规避

一、背景 分享一个在项目运维中遇到的一个主从复制限制的一个坑&#xff0c;项目的架构为主集群灾备集群&#xff0c;每个集群为一主两从模式。主集群到灾备集群的同步为主从复制的方式&#xff0c;根据业务需求灾备集群需要忽略系统库跟某些配置表&#xff0c;所以才会触发此…

24v转12v转9v转5v转4.2v降压电源芯片AH8788

AH8788A是一款集成同步开关的降压转换器&#xff0c;提供***解决方案适用于车载充电器、快充适配器和智能排插。AH8788A内置功率MOS&#xff0c;输入电压范围为9.6V到32V&#xff0c;输出电压范围为3V到12V&#xff0c;***-大可提供18W的输出功率。它能够根据识别到的快充协议自…

手把手教你用 Milvus 和 Towhee 搭建一个 AI 聊天机器人!

作为向量数据库的佼佼者&#xff0c;Milvus 适用于各种需要借助高效和可扩展向量搜索功能的 AI 应用。 举个例子&#xff0c;如果想要搭建一个负责聊天机器人数据管理流程&#xff0c;Milvus 必然是首选向量数据库。那么如何让这个应用程序开发变得易于管理及更好理解&#xff…

速看!美国站新增1个禁售品类,加拿大站3大品类开启售前审核

亚马逊新合规 美国&加拿大◀ 一品类禁售&#xff0c;三品类售前审核 近日&#xff0c;亚马逊发布了合规政策的新要求&#xff0c;其中美国站“呼吸贴”被归类为禁售的产品&#xff0c;加拿大站“儿童床垫”、“夜灯”、“儿童折叠式和非折叠式椅子和凳子”品类均有合规要…

家政服务小程序,家政系统开发

家政服务小程序&#xff0c;家政系统开发&#xff0c;打造一线家政系统&#xff0c;提效增收 家政服务小程序 互联网&#xff0b;家政系统&#xff0c;打造互联网&#xff0b;家政公司app开发&#xff0c;支持个性化定制&#xff0c;直接搭建&#xff0c;上手即用&#xff0e;实…

Redis〔篇〕

redis怎么做到双写一致性呢&#xff1f; 这个是要分情况的 业务要是对一致性要求不是很高的话可以使用延时双删&#xff0c;要强一致的话需要双写一致性。 Redis数据持久化&#xff1f; redis是有两种数据持久化方式的&#xff0c;一种RDB一种AOF rdb是redis数据快照&#x…

大型DOM结构是如何影响交互性的

没有办法绕过这一点&#xff1a;当你构建一个网页时&#xff0c;该页面一定会有一个文档对象模型&#xff08;DOM&#xff09;。DOM代表了你页面HTML的结构&#xff0c;并为JavaScript和CSS提供了访问页面结构和内容的途径。 然而&#xff0c;问题在于DOM的大小会影响浏览器快速…

新手科普!UX设计师是做什么的?

什么是UX设计师&#xff1f; UX设计师(User Experience Designer)&#xff0c;又称用户体验设计师&#xff0c;顾名思义UX设计师是负责设计产品/服务的用户体验的专业人员。UX设计师涵盖了用户调研、交互设计、原型设计、动效设计、UI设计等工作内容。 本文主要介绍数字化软件…

Python机器学习实战-特征重要性分析方法(5):递归特征消除(附源码和实现效果)

实现功能 递归地删除特征并查看它如何影响模型性能。删除时会导致更大下降的特征更重要。 实现代码 from sklearn.ensemble import RandomForestClassifier from sklearn.feature_selection import RFE import pandas as pd from sklearn.datasets import load_breast_cance…

【C语言】【结构体的内存对齐】计算结构体内存大小,有图解

计算结构体内存大小&#xff0c;需要用到结构体内存对齐的知识 来段代码看看什么是结构体对齐&#xff1a; #include<stdio.h> struct S1 {char a;char b;int num; }; struct S2 {char a;int num;char b; }; int main() {printf("%zd\n", sizeof(struct S1))…

真·Redis缓存优化—97%的优化率你见过嘛? | 京东云技术团队

本文通过一封618前的R2M(公司内部缓存组件&#xff0c;可以认为等同于Redis)告警&#xff0c;由浅入深的分析了该告警的直接原因与根本原因&#xff0c;并根据原因提出相应的解决方法&#xff0c;希望能够给大家在排查类似问题时提供相应的思路。 一、问题排查 1.1 邮件告警 …

如何看待Java上层技术与JVM

如何看待Java上层技术与JVM 你是否也遇到过这些问题 运行着的线上系统突然卡死&#xff0c;系统无法访问&#xff0c;甚至直接OOM&#xff01;想解决线上JVM GC问题&#xff0c;但却无从下手。新项目上线&#xff0c;对各种JVM参数设置一脸茫然&#xff0c;直接默认吧&#x…

景联文数据标注:ChatGPT成功的秘密——人类反馈强化学习(RLHF)

ChatGPT的成功很大程度上归功于其采用的新的训练范式——人类反馈强化学习&#xff08;RLHF&#xff09;。RLHF是一种强化学习方法&#xff0c;它将强化学习与人类反馈相结合&#xff0c;通过利用人类提供的反馈来指导智能系统的行为&#xff0c;使其能够更加高效、快速地学习任…

分布式锁工具Redisson(Lua脚本)

如何实现分布式锁&#xff1f; Redis 可以通过 setnx&#xff08;set if not exists&#xff09;命令实现分布式锁 通过执行结果是否为 1 可以判断是否成功获取到锁 setnx mylock true 加锁del mylock 释放锁 分布式锁存在的问题&#xff1a; 死锁问题&#xff0c;未设置过…

运行在浏览器中的Domino Designer开发客户机

大家好&#xff0c;才是真的好。 首先讨论一个非常有意思的事情&#xff0c;就是有人问&#xff0c;如果我用很老的Lotus软件&#xff0c;它是免费的吗&#xff1f; 这估计代表了很多盆友的心声。但不太友好的是&#xff0c;即使你用很老的Lotus软件&#xff08;例如Notes R4…