热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Leetcode题解动态规划01背包(1):问题简介

0-1背包有一个容量为N的背包,要用这个背包装下物品的价值最大,这些物品有两个属性:体积w和价值v。定义一个二维数组dp存储最大价值&
0-1 背包

有一个容量为 N 的背包,要用这个背包装下物品的价值最大,这些物品有两个属性:体积 w 和价值 v。

定义一个二维数组 dp 存储最大价值,其中 dp[i][j] 表示前 i 件物品体积不超过 j 的情况下能达到的最大价值。设第 i 件物品体积为 w,价值为 v,根据第 i 件物品是否添加到背包中,可以分两种情况讨论:

  • 第 i 件物品没添加到背包,总体积不超过 j 的前 i 件物品的最大价值就是总体积不超过 j 的前 i-1 件物品的最大价值,dp[i][j] = dp[i-1][j]。
  • 第 i 件物品添加到背包中,dp[i][j] = dp[i-1][j-w] + v。

第 i 件物品可添加也可以不添加,取决于哪种情况下最大价值更大。因此,0-1 背包的状态转移方程为:

 

dp[i][j] = value dp[i - 1][j - w] + v;

public int knapsack(int W, int N, int[] weights, int[] values) {int[][] dp &#61; new int[N &#43; 1][W &#43; 1];for (int i &#61; 1; i <&#61; N; i&#43;&#43;) {int w &#61; weights[i - 1], v &#61; values[i - 1];for (int j &#61; 1; j <&#61; W; j&#43;&#43;) {if (j >&#61; w) {dp[i][j] &#61; Math.max(dp[i - 1][j], dp[i - 1][j - w] &#43; v);} else {dp[i][j] &#61; dp[i - 1][j];}}}return dp[N][W];
}

public int knapsack(int W, int N, int[] weights, int[] values){int[][] dp &#61; new int[N&#43;1][W&#43;1];for(int i&#61; 1; i<&#61;N; i&#43;&#43;){int w &#61; weights[i-1], v&#61; vlaues[i-1];for(int j&#61;1; j<&#61;W; j&#43;&#43;){if(j >&#61; w){dp[i][j] &#61; Math.max(dp[i-1][j], dp[i-1][j-w] &#43; v);elsedp[i][j] &#61; dp[i-1][j];}}}return dp[N][W];
}

空间优化

在程序实现时可以对 0-1 背包做优化。观察状态转移方程可以知道&#xff0c;前 i 件物品的状态仅与前 i-1 件物品的状态有关&#xff0c;因此可以将 dp 定义为一维数组&#xff0c;其中 dp[j] 既可以表示 dp[i-1][j] 也可以表示 dp[i][j]。此时&#xff0c;

 

因为 dp[j-w] 表示 dp[i-1][j-w]&#xff0c;因此不能先求 dp[i][j-w]&#xff0c;以防将 dp[i-1][j-w] 覆盖。也就是说要先计算 dp[i][j] 再计算 dp[i][j-w]&#xff0c;在程序实现时需要按倒序来循环求解。

public int knapsack(int W, int N, int[] weights, int[] values) {int[] dp &#61; new int[W &#43; 1];for (int i &#61; 1; i <&#61; N; i&#43;&#43;) {int w &#61; weights[i - 1], v &#61; values[i - 1];for (int j &#61; W; j >&#61; 1; j--) {if (j >&#61; w) {dp[j] &#61; Math.max(dp[j], dp[j - w] &#43; v);}}}return dp[W];
}

public int knapsack(int W, int N, int[] weights, int[] values){int[] dp &#61; new int[W&#43;1];for(int i&#61;1; i<&#61;N; i&#43;&#43;){int w &#61; weights[i-1], v&#61; values[i-1];for(int j&#61;W; j>&#61;1; j--){if(j >&#61; w)dp[j] &#61; Math.max(dp[j], dp[j-w] &#43; v);}}
}

无法使用贪心算法的解释

0-1 背包问题无法使用贪心算法来求解&#xff0c;也就是说不能按照先添加性价比最高的物品来达到最优&#xff0c;这是因为这种方式可能造成背包空间的浪费&#xff0c;从而无法达到最优。考虑下面的物品和一个容量为 5 的背包&#xff0c;如果先添加物品 0 再添加物品 1&#xff0c;那么只能存放的价值为 16&#xff0c;浪费了大小为 2 的空间。最优的方式是存放物品 1 和物品 2&#xff0c;价值为 22.

idwvv/w
0166
12105
23124

变种


  • 完全背包&#xff1a;物品数量为无限个

  • 多重背包&#xff1a;物品数量有限制

  • 多维费用背包&#xff1a;物品不仅有重量&#xff0c;还有体积&#xff0c;同时考虑这两种限制

  • 其它&#xff1a;物品之间相互约束或者依赖


推荐阅读
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • C++实现经典排序算法
    本文详细介绍了七种经典的排序算法及其性能分析。每种算法的平均、最坏和最好情况的时间复杂度、辅助空间需求以及稳定性都被列出,帮助读者全面了解这些排序方法的特点。 ... [详细]
  • 深入理解C++中的KMP算法:高效字符串匹配的利器
    本文详细介绍C++中实现KMP算法的方法,探讨其在字符串匹配问题上的优势。通过对比暴力匹配(BF)算法,展示KMP算法如何利用前缀表优化匹配过程,显著提升效率。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 在金融和会计领域,准确无误地填写票据和结算凭证至关重要。这些文件不仅是支付结算和现金收付的重要依据,还直接关系到交易的安全性和准确性。本文介绍了一种使用C语言实现小写金额转换为大写金额的方法,确保数据的标准化和规范化。 ... [详细]
  • 使用Numpy实现无外部库依赖的双线性插值图像缩放
    本文介绍如何仅使用Numpy库,通过双线性插值方法实现图像的高效缩放,避免了对OpenCV等图像处理库的依赖。文中详细解释了算法原理,并提供了完整的代码示例。 ... [详细]
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本题探讨了一种字符串变换方法,旨在判断两个给定的字符串是否可以通过特定的字母替换和位置交换操作相互转换。核心在于找到这些变换中的不变量,从而确定转换的可能性。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • 本文将介绍如何使用 Go 语言编写和运行一个简单的“Hello, World!”程序。内容涵盖开发环境配置、代码结构解析及执行步骤。 ... [详细]
  • 本文探讨了Hive中内部表和外部表的区别及其在HDFS上的路径映射,详细解释了两者的创建、加载及删除操作,并提供了查看表详细信息的方法。通过对比这两种表类型,帮助读者理解如何更好地管理和保护数据。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文详细探讨了Java中的24种设计模式及其应用,并介绍了七大面向对象设计原则。通过创建型、结构型和行为型模式的分类,帮助开发者更好地理解和应用这些模式,提升代码质量和可维护性。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
author-avatar
风让我离开
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有