【随想录】Day44—第九章 动态规划part06

2024-05-12 01:52

本文主要是介绍【随想录】Day44—第九章 动态规划part06,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这里写目录标题

  • 题目1: 完全背包
    • 1- 什么是完全背包问题?
      • 先遍历物品 和 先遍历背包的区别
      • 动规五部曲
    • 2- 题解
      • ⭐ 完全背包——题解思路
  • 题目2: 518. 零钱兑换 II
    • 1- 思路
      • 动规五部曲
    • 2- 题解
      • ⭐零钱兑换 II ——题解思路
    • 题目3: 377. 组合总和 Ⅳ
      • 1- 思路
        • 动规五部曲
    • 2- 题解
      • ⭐组合总和 Ⅳ——题解思路


题目1: 完全背包

  • 题目链接:52. 携带研究材料

1- 什么是完全背包问题?

  • 01背包的核心代码
for(int i = 0; i < weight.size(); i++) { // 遍历物品for(int j = bagWeight; j >= weight[i]; j--) { // 遍历背包容量dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);}
}
  • 在完全背包中如果希望物品使用无限次,此时需要修改遍历的顺序,正序遍历即可实现每个物品使用无限次。
  • 我们知道 01背包 内嵌的循环是从大到小遍历,为了保证每个物品仅被添加一次。
  • 而完全背包的物品是可以添加多次的,所以要从小到大去遍历,即:
// 先遍历物品,再遍历背包
for(int i = 0; i < weight.size(); i++) { // 遍历物品for(int j = weight[i]; j <= bagWeight ; j++) { // 遍历背包容量dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);}
}
  • dp状态图如下:


相信很多同学看网上的文章,关于完全背包介绍基本就到为止了。
其实还有一个很重要的问题,为什么遍历物品在外层循环,遍历背包容量在内层循环?
这个问题很多题解关于这里都是轻描淡写就略过了,大家都默认 遍历物品在外层,遍历背包容量在内层,好像本应该如此一样,那么为什么呢?
难道就不能遍历背包容量在外层,遍历物品在内层?


先遍历物品 和 先遍历背包的区别

  • 1. 先遍历物品 再 遍历背包
  • 对于 coins = {1,2,5} 此时最后的结果只能出现 {1,2} 不会出现 {2,1} 的情况
  • 2. 先遍历背包 再 遍历物品
  • 最终得到的是排列数

动规五部曲

  • 1. 定义 dp 数组以及其含义
    • dp[i] 代表,容量为 i 的背包的最大效益
  • 2. 递推公式
    • dp[j] = Math.max(dp[j],dp[j-weight[i]]+value[i]);
  • 3. dp 数组初始化
    • dp[0] = 0
  • 4.确定遍历顺序
    • ① 先遍历物品i0 遍历到 物品的个数 n
    • ② 后遍历背包j 从背包重量 weight[i] 遍历到 v

2- 题解

⭐ 完全背包——题解思路

在这里插入图片描述

import java.util.Scanner;
public class Main {public static void main(String[] args) {int n = 0;int v = 0;Scanner sc = new Scanner(System.in);n = sc.nextInt();  // 物品v = sc.nextInt();  // 背包重量// 初始化 weight 数组 和 value 数组int[] weight = new int[n];int[] value = new int[n];for(int i = 0 ; i < n;i++){weight[i] = sc.nextInt();value[i] = sc.nextInt();}// 1. 定义 dp 数组int[] dp = new int[v+1];// 2. 确定递推公式// dp[j] = Math.max(dp[j],dp[j-weight[i]]+value[i]);// 3. 初始化dp[0] = 0;for(int i = 0 ; i < n;i++){ // 遍历物品for(int j = weight[i]; j <= v ;j++){dp[j] = Math.max(dp[j],dp[j-weight[i]]+value[i]);}}System.out.println(dp[v]);}
}

题目2: 518. 零钱兑换 II

  • 题目链接:518. 零钱兑换 II

1- 思路

动规五部曲

  • 1. 确定 dp 数组含义
    • dp[j]:凑成总金额j的货币组合数为dp[j]
  • 2. 确定递推公式
    • dp[j] += dp[j-coins[i]]
  • 3. dp 数组初始化
    • dp[0] = 1
  • 4. 遍历顺序
    • ① 先遍历物品i0 遍历到 物品的个数 coins.length
    • ② 后遍历背包j 从背包重量 coins[i] 遍历到 amount


2- 题解

⭐零钱兑换 II ——题解思路

在这里插入图片描述

class Solution {public int change(int amount, int[] coins) {// 维度 1 target = amount// 维度 2 面额 = coins// 目的:根据面额凑出 target // 1. 定义 dp 数组 //  达到 target 需要的组合种类数 int[] dp = new int[amount+1];// 2. 确定递推公式// 3. 初始化dp[0] = 1;// 4. 遍历顺序// 先物品后背包for(int i = 0; i < coins.length ;i++){for(int j = coins[i]; j <= amount ;j++){dp[j] += dp[j-coins[i]];}}return dp[amount];}
}

题目3: 377. 组合总和 Ⅳ

  • 题目链接:377. 组合总和 Ⅳ

本题目和零钱兑换 II 的区别:

  • 零钱兑换不强调元素的顺序,即 1,1,2 和 1,2,1 实际上是按照一种情况来算
  • 但在本题目中, 1,1,2 和 1,2,1 需要进行分别计算

注意:

  • ① 先遍历物品再遍历背包: 得到的是组合数
  • ② 先遍历背包再遍历物品:得到的是排列数

1- 思路

动规五部曲
  • 1. 确定 dp 数组含义
    • dp[j]:凑成总金额j的货币组合数为dp[j]
  • 2. 确定递推公式
    • dp[j] += dp[j-coins[i]]
  • 3. dp 数组初始化
    • dp[0] = 1
  • 4. 遍历顺序(所求为组合,因此需要先遍历背包再遍历物品)
    • 背包:i 从 0 遍历到 target ,即for(int i = 0 ; i <=target ; i++)
    • 物品:j 从 0 遍历到 nums.length ,即 for(int j = 0 ; j <= nums.length; j++)

2- 题解

⭐组合总和 Ⅳ——题解思路

在这里插入图片描述

class Solution {public int combinationSum4(int[] nums, int target) {// 物品: nums.length// 背包: targetint n = nums.length;// 1. 定义 dp 数组int[] dp = new int[target+1];// 2. 递推公式// dp[j] += dp[j-weight[i]]// 3. 初始化dp[0] = 1;// 4.遍历顺序// 遍历 物品// 遍历 背包for (int i = 0; i <= target; i++) {for (int j = 0; j < nums.length; j++) {if (i >= nums[j]) {dp[i] += dp[i - nums[j]];}}}return dp[target];}
}

这篇关于【随想录】Day44—第九章 动态规划part06的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/981271

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

Java使用POI-TL和JFreeChart动态生成Word报告

《Java使用POI-TL和JFreeChart动态生成Word报告》本文介绍了使用POI-TL和JFreeChart生成包含动态数据和图表的Word报告的方法,并分享了实际开发中的踩坑经验,通过代码... 目录前言一、需求背景二、方案分析三、 POI-TL + JFreeChart 实现3.1 Maven

Java导出Excel动态表头的示例详解

《Java导出Excel动态表头的示例详解》这篇文章主要为大家详细介绍了Java导出Excel动态表头的相关知识,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录前言一、效果展示二、代码实现1.固定头实体类2.动态头实现3.导出动态头前言本文只记录大致思路以及做法,代码不进

vue基于ElementUI动态设置表格高度的3种方法

《vue基于ElementUI动态设置表格高度的3种方法》ElementUI+vue动态设置表格高度的几种方法,抛砖引玉,还有其它方法动态设置表格高度,大家可以开动脑筋... 方法一、css + js的形式这个方法需要在表格外层设置一个div,原理是将表格的高度设置成外层div的高度,所以外层的div需要

SpringBoot实现动态插拔的AOP的完整案例

《SpringBoot实现动态插拔的AOP的完整案例》在现代软件开发中,面向切面编程(AOP)是一种非常重要的技术,能够有效实现日志记录、安全控制、性能监控等横切关注点的分离,在传统的AOP实现中,切... 目录引言一、AOP 概述1.1 什么是 AOP1.2 AOP 的典型应用场景1.3 为什么需要动态插

VUE动态绑定class类的三种常用方式及适用场景详解

《VUE动态绑定class类的三种常用方式及适用场景详解》文章介绍了在实际开发中动态绑定class的三种常见情况及其解决方案,包括根据不同的返回值渲染不同的class样式、给模块添加基础样式以及根据设... 目录前言1.动态选择class样式(对象添加:情景一)2.动态添加一个class样式(字符串添加:情

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

如何用Python绘制简易动态圣诞树

《如何用Python绘制简易动态圣诞树》这篇文章主要给大家介绍了关于如何用Python绘制简易动态圣诞树,文中讲解了如何通过编写代码来实现特定的效果,包括代码的编写技巧和效果的展示,需要的朋友可以参考... 目录代码:效果:总结 代码:import randomimport timefrom math

Java中JSON字符串反序列化(动态泛型)

《Java中JSON字符串反序列化(动态泛型)》文章讨论了在定时任务中使用反射调用目标对象时处理动态参数的问题,通过将方法参数存储为JSON字符串并进行反序列化,可以实现动态调用,然而,这种方式容易导... 需求:定时任务扫描,反射调用目标对象,但是,方法的传参不是固定的。方案一:将方法参数存成jsON字

.NET利用C#字节流动态操作Excel文件

《.NET利用C#字节流动态操作Excel文件》在.NET开发中,通过字节流动态操作Excel文件提供了一种高效且灵活的方式处理数据,本文将演示如何在.NET平台使用C#通过字节流创建,读取,编辑及保... 目录用C#创建并保存Excel工作簿为字节流用C#通过字节流直接读取Excel文件数据用C#通过字节