[笔记]动态规划之01背包问题

2024-09-04 04:58

本文主要是介绍[笔记]动态规划之01背包问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

01背包

  • n件物品
  • w为容量上限的背包
  • weight[i]:第i件物品的重量
  • value[i]:第i件物品的价值

每件物品只能放入1次,装入哪些物品能让背包内物品总价值最高?

暴力解法:回溯算法 - 时间复杂度 O ( 2 n ) O(2^n) O(2n)

二维数组

  1. 确定dp数组下标含义

    dp[i][j] - 从下标为[0]到[i]的物品里取,放入容量为j的背包,此时的最大价值总和

  2. 确定递推公式

    • 不放物品i:dp[i][j] = dp[i - 1][j](物品重量大于背包容量,物品i放不进包里,背包价值不变;
    • 放物品i:dp[i][j] = dp[i - 1][j - weight[i]] + value[i]dp[i - 1][j - weight[i]]是容量为j - weight[i]时未放物品i的最大价值,dp[i - 1][j - weight[i]] + value[i]是此时放入物品i后背包的最大价值;
    • 综上:dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i])
  3. 初始化dp数组

    1. 容量j为0时,背包价值总和一定为0;

    2. 当j < weight[0]时,dp[0][j] = 0(背包容量比第0号物品小,放不进背包);

    3. 当j > weight[0]时,dp[0][j] = value[0](背包可以放下第0号物品);

      //参考代码如下,全部初始化为0则可省略第一个for循环
      for (int j = 0; j < weight[0]; j++)dp[0][j] = 0;
      for (int j = weight[0]; j <= bagweight; j++) dp[0][j] = value[0];
      
  4. 确定遍历顺序

    1. 先遍历物品,后遍历背包
    2. 先遍历背包,后遍历物品
    3. 本质上:dp[i][j]由前序数据推出,for循环遍历次序不影响公式推导
  5. 举例推导dp数组

滚动数组

所谓滚动数组,就是通过一维数组的方式压缩存储背包问题的状态。

  1. 确定dp数组下标含义

    • dp[i][j] - 从下标为[0]到[i]的物品里取,放入容量为j的背包,此时的最大价值总和 ;
    • dp[j] - 容量为j的背包,所装物品的最大价值总和;
  2. 确定dp数组递推公式

    1. 不放物品i:dp[j] = dp[j]
    2. 放物品i:dp[j] = dp[j - weight[i]] + value[i]
    3. 综上,递推式为dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);
  3. 初始化dp数组

    • dp[0] = 0;
    • 其它下标可以初始化为0;
  4. 确定遍历顺序

    • 倒序遍历:保证物品i只被放入一次!

      dp[i][j]是由上一层dp[i - 1][ ]计算而来,当前层的dp[i][j]并不会被覆盖,顺序逆序遍历物品i都只被放入一次。如果顺序遍历dp[j]数组,就相当于dp[i][j]由层dp[i][ ]计算而来,这个过程就重复放入物品i了。

    • 先遍历物品,再嵌套遍历背包容量

      倒序遍历,如果先遍历背包容量,再嵌套物品,那么背包里都只放入了一个物品。

  5. 举例推导dp数组

这篇关于[笔记]动态规划之01背包问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

Vue3绑定props默认值问题

《Vue3绑定props默认值问题》使用Vue3的defineProps配合TypeScript的interface定义props类型,并通过withDefaults设置默认值,使组件能安全访问传入的... 目录前言步骤步骤1:使用 defineProps 定义 Props步骤2:设置默认值总结前言使用T

Web服务器-Nginx-高并发问题

《Web服务器-Nginx-高并发问题》Nginx通过事件驱动、I/O多路复用和异步非阻塞技术高效处理高并发,结合动静分离和限流策略,提升性能与稳定性... 目录前言一、架构1. 原生多进程架构2. 事件驱动模型3. IO多路复用4. 异步非阻塞 I/O5. Nginx高并发配置实战二、动静分离1. 职责2

解决升级JDK报错:module java.base does not“opens java.lang.reflect“to unnamed module问题

《解决升级JDK报错:modulejava.basedoesnot“opensjava.lang.reflect“tounnamedmodule问题》SpringBoot启动错误源于Jav... 目录问题描述原因分析解决方案总结问题描述启动sprintboot时报以下错误原因分析编程异js常是由Ja

MySQL 表空却 ibd 文件过大的问题及解决方法

《MySQL表空却ibd文件过大的问题及解决方法》本文给大家介绍MySQL表空却ibd文件过大的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录一、问题背景:表空却 “吃满” 磁盘的怪事二、问题复现:一步步编程还原异常场景1. 准备测试源表与数据

解决Nginx启动报错Job for nginx.service failed because the control process exited with error code问题

《解决Nginx启动报错Jobfornginx.servicefailedbecausethecontrolprocessexitedwitherrorcode问题》Nginx启... 目录一、报错如下二、解决原因三、解决方式总结一、报错如下Job for nginx.service failed bec

SysMain服务可以关吗? 解决SysMain服务导致的高CPU使用率问题

《SysMain服务可以关吗?解决SysMain服务导致的高CPU使用率问题》SysMain服务是超级预读取,该服务会记录您打开应用程序的模式,并预先将它们加载到内存中以节省时间,但它可能占用大量... 在使用电脑的过程中,CPU使用率居高不下是许多用户都遇到过的问题,其中名为SysMain的服务往往是罪魁

MySQ中出现幻读问题的解决过程

《MySQ中出现幻读问题的解决过程》文章解析MySQLInnoDB通过MVCC与间隙锁机制在可重复读隔离级别下解决幻读,确保事务一致性,同时指出性能影响及乐观锁等替代方案,帮助开发者优化数据库应用... 目录一、幻读的准确定义与核心特征幻读 vs 不可重复读二、mysql隔离级别深度解析各隔离级别的实现差异

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

Python多线程应用中的卡死问题优化方案指南

《Python多线程应用中的卡死问题优化方案指南》在利用Python语言开发某查询软件时,遇到了点击搜索按钮后软件卡死的问题,本文将简单分析一下出现的原因以及对应的优化方案,希望对大家有所帮助... 目录问题描述优化方案1. 网络请求优化2. 多线程架构优化3. 全局异常处理4. 配置管理优化优化效果1.