九章算法面试题58 背包问题

九章算法官网-原文网址

http://www.jiuzhang.com/problem/58/

题目

有一个大小为m(整数)的背包,和n个体积为正整数的物品(大小分别为A[i])。将这个n个物品选一些装到背包中,请问最多能装满多少的体积?

在线测试本题

http://www.lintcode.com/problem/backpack/

解答

背包问题是动态规划问题的一种典型题目。 动态规划问题我们一般要考虑下面这四点。

1. 状态 State

2. 方程 Function

3. 初始化 Intialization

4. 答案 Answer

本题是最基础的背包问题,特点是:每种物品仅有一件,可以选择放或不放,那下面我们来看背包这题动态规划的四点是怎么样的呢?

1. State: dp[i][S] 表示前i个物品,取出一些能否组成和为S体积的背包

2. Function: f[i][S] = f[i-1][S - A[i]] or f[i-1][S] (A[i]表示第i个物品的大小)

转移方程想得到f[i][S]前i个物品取出一些物品想组成S体积的背包。 那么可以从两个状态转换得到。

(1)f[i-1][S - A[i]] 放入第i个物品,并且前i-1个物品能否取出一些组成和为S-A[i] 体积大小的背包。

(2)f[i-1][S] 不放入第i个物品, 并且前i-1个物品能否取出一些组成和为S 体积大小的背包。

3. Intialize: f[1...n][0] = true; f[0][1... m] = false

初始化 f[1...n][0] 表示前1...n个物品,取出一些能否组成和为0 大小的背包始终为真。

其他初始化为假

4. Answer: 寻找使f[n][S] 值为true的最大的S. (S的取值范围1到m)

由于这道题空间上有一些要求,所以在知道了思路答案过后我们还需要进行优化空间复杂度.先考虑上面讲的基本思路如何实现,肯定是有一个主循环i=1..N,每次算出来二维数组f[i][0..S]的所有值。那么,如果只用一个数组f[0..S],能不能保证第i次循环结束后f[S]中表示的就是我们定义的状态f[i][S]呢?f[i][S]是由 f[i-1][S - a[i]] 和 f[i-1][S]
两个子问题递推而来,能否保证在推f[i][S]时(也即在第i次主循环中推f[S]时)能够得到 f[i-1][S - a[i]] 和 f[i-1][S] 的值呢?事实上,这要求在每次主循环中我们以S=m...0的顺序推f[S],这样才能保证推f[S]时f[S-a[i]]保存的是状态f[i-1][S-a[i]]的值。伪代码如下:

for i=1..N

for S=m...0

f[S]=f[S] or f[S-a[i]];

时间: 2024-10-17 00:23:19

九章算法面试题58 背包问题的相关文章

九章算法面试题59 背包问题II

九章算法官网-原文网址 http://www.jiuzhang.com/problem/59/ 题目 有一个大小为m(整数)的背包,和n个体积(大小分别为A[i](整数))和价值(价格分别为B[i](整数))的物品.将这n个物品选一些装到背包中,请问能装价值最大为多少的物品? 在线测试本题 http://www.lintcode.com/en/problem/backpack-ii/ 解答 设F[i][j]为前i个物品,选一些组成j的体积,能够获得最大的价值是多少.状态转移方程:f[i][j]

九章算法面试题43 直方图内最大矩阵

九章算法官网-原文网址 http://www.jiuzhang.com/problem/43/ 题目 给出一个直方图(如图所示),求出所给直方图中所包含的最大矩阵面积.直方图可以用一个整数数组表示,如上图为[2, 1, 5, 6, 2, 3].每个直方块的宽度均为1.上图中包含的最大矩阵面积为10. 解答 如果对于每个直方块,找到从它开始往左边数第一个比它小的,和往右边数第一个比他小的,则可以确定出以该直方块为最矮一块的矩阵的最大面积.使用数据结构栈,栈中保存递增序列,从左到右依次遍历每个数让其

九章算法面试题31 子数组的最大差

九章算法官网-原文网址 http://www.jiuzhang.com/problem/31/ 题目 给定一个数组,求两个不相交的并且是连续的子数组A和B(位置连续),满足|sum(A) - sum(B)|最大(和之差的绝对值).例如[2, -1, -2, 1, -4, 2, 8],可以得到A=[-1, -2, 1, -4], B=[2, 8],最大差为16. 解答 预处理每个位置往左/右的最大/最小子数组,然后再枚举划分位置,求得所有MaxLeft[i] - MinRight[i+1]和Max

九章算法面试题37 主元素

九章算法官网-原文网址 http://www.jiuzhang.com/problem/37/ 题目 主元素(Majority Number)定义为数组中出现次数严格超过一半的数.找到这个数.要求使用O(1)的额外空间和O(n)的时间. 进阶1:如果数组中存在且只存在一个出现次数严格超过1/3的数,找到这个数.要求使用O(1)的额外空间和O(n)的时间. 进阶2:如果数组中存在且只存在一个出现次数严格超过1/k的数,找到这个数.要求使用O(k)的额外空间和O(n)的时间 解答 采用抵消法.一旦发

九章算法面试题40 不用除法求积

九章算法官网-原文网址 http://www.jiuzhang.com/problem/40/ 题目 给定一个数组A[1..n],求数组B[1..n],使得B[i] = A[1] * A[2] .. * A[i-1] * A[i+1] .. * A[n].要求不要使用除法,且在O(n)的时间内完成,使用O(1)的额外空间(不包含B数组所占空间). 解答 计算前缀乘积Prefix[i] = A[1] * A[2] .. A[i],计算后缀乘积Suffix[i] = A[i] * A[i+1] ..

九章算法面试题50 队列上实现Min函数

九章算法官网-原文网址 http://www.jiuzhang.com/problem/50/ 题目 ?在<九章算法面试题23 栈上实现Min函数>中,我们介绍了在栈上实现一个O(1)的Min方法.那么,如何在队列上实现一个Min方法? 要求,队列除了支持基本的Push(x) Pop()的方法以外,还需要支持Min方法,返回当前队列中的最小元素.每个方法的均摊复杂度为O(1) 解答 在九章面试题49<用栈实现队列>和面试题23<栈上实现Min函数>中,我们讲解到了如何用

九章算法面试题42 构造MaxTree

九章算法官网-原文网址 http://www.jiuzhang.com/problem/42/ 题目 给定一个没有重复元素的数组A,定义A上的MaxTree如下:MaxTree的根节点为A中最大的数,根节点的左子树为数组中最大数左边部分的MaxTree,右子树为数组中最大数右边部分的MaxTree.请根据给定的数组A,设计一个算法构造这个数组的MaxTree. 解答 如果能够确定每个节点的父亲节点,则可以构造出整棵树.找出每个数往左数第一个比他大的数和往右数第一个比他大的数,两者中较小的数即为该

九章算法面试题36 交错的字符串

九章算法官网-原文网址 http://www.jiuzhang.com/problem/36/ 题目 给定三个字符串A, B, C,判断C是否由A和B交错构成.交错构成的意思是,对于字符串C,可以将其每个字符标记为A类或B类,使得我A类的每个字符顺序构成了A字符串,B类的每个字符顺序构成了B字符串.如:对于A="rabbit" B="mq", "rabmbitq"是由A和B交错构成的,但"rabbqbitm"不是由A和B交错构

九章算法面试题45 寻找最大的储水容器

九章算法官网-原文网址 http://www.jiuzhang.com/problem/45/ 题目 给定一个正整数数组(a0,a1..),分别代表n个坐标(0,a0), (1,a1),根据这n个点画出n条线段,每条线段的两个端点分别为(i, ai)和(i, 0).找到两条线段,使得这两条线段和x轴所构成的容器储水容量最大.如[2,1,3], 最大,选择第一条线段和第三条线段,加上x轴所构成的容器,储水容量为4(高度为Min(2,3)=2,底为2) 解答 用两根指针,一根指向头,一根指向尾,如果