动态规划之背包问题

  背包问题是一个经典的算法问题,可以用动态规划,贪心法,分支界限法等方法解决。问题描述:有n个物品,编号1,2,3,、、n,其中第 i 个物品重量为Wi 价值 Vi ,有一个容量为W的背包。在容量允许范围内,如何选择物品,可以得到最大的价值。(为了简单起见,假设物品的重量 Wi 和价值Vi 都是正数)

  今天主要说的是0、1背包问题,解法是动态规划。当然,对于另外两种问题也会有所介绍。

问题分析:

  用动态规划解问题首先要有效的找出子问题,可以通过这个子问题得推得原问题的解,通常子问题的实质是和原问题相同的,只是规模上的缩小,也就是说子问题和原问题可以有相同的表示形式,问题可通过不断的缩小规模(一般都会有一个界限)能找到子问题的解。

  这个问题要求解的是能用背包带走的物品的最大价值。定义 m[i,w] 为:用第1,、2、3、、i 个物品装入质量<=W的背包的最大价值。

  m[i,w]的取值情况分析:

    1)  ,背包的质量为w,里面没有物品,所以它的价值为0;

    2)   ,背包质量为0,所以里面没法装任何东西, 不论前面的 i 是多少,总价值为0;

  对于任意的第 i 个物品,有两种情况,放进背包或者不放。不要第 i 个物品 如果 则:

    3)因为第i 个物品的重量大于背包的容量,所以不可放入。

  如果. 那么

    4)

  对于第 i 个物品,有两种可选择方案:如果放入背包中,那么 m[i,w]=m[i-1,w-wi]+vi。也就是i的前一个的最大值加上自己的价值。如果不要第i个物品,那么:m[i,w]=m[i-1,w]。也就是 i 的前一个的最大值。因为背包问题最后要取得最大的价值,所以就选这两种情况中价值最大的。

  在这个问题中,定义子问题: m[i,w] 对于每个子问题,都可通过上面的分析求出。通过3),4)可以发现,每一次求取子问题,问题的规模就被缩小。要么在w 上减小,要么在 i 上减小。最后问题的规模会被缩小为 m[i,0]和m[0,w].而这两个的值都为0,只要逆向思维反推回去,就能逐步得到问题的解。

算法描述

算法实现JAVA版

public class Knapsack {
    static int totalWeight=10;//背包的容量
    static int[] w=new int[]{0,2,2,6,5,4};//物品的总重量,其中0号位不使用
    static int[] v=new int[]{0,6,3,5,4,6};
    static int[] x=new int[w.length];
    static int[][] matri=new int[w.length][totalWeight+1];
    public static void package0_1(){
        for(int j=0;j<=totalWeight;j++){
            matri[0][j]=0;//第一行初始为0
        }
        for(int j=1;j<=v.length-1;j++){
            matri[j][0]=0;//第一列初始为0
        }

        for (int i=1; i <=w.length-1; i++) {
            for (int j =1 ; j<=totalWeight; j++) {
                if(j<w[i]){
                    matri[i][j]=matri[i-1][j];
                }else{
                    matri[i][j]=Math.max(matri[i-1][j],matri[i-1][j-w[i]]+v[i]);
                }
            }
        }
    }

    public static void answer(){
        int j=totalWeight;
        int i;
        for (i=w.length-1; i>=1;i--) {
            if(matri[i][j]==matri[i-1][j]){
                x[i]=0;
            }else{
                x[i]=1;
                j-=w[i];
            }
        }
        x[0]=0;
    }
    public static void main(String[] args) {
        package0_1();
        answer();
        for (int i = 0; i < x.length; i++) {
            System.out.println(x[i]);
        }
    }
}
时间: 2024-08-11 02:34:24

动态规划之背包问题的相关文章

动态规划:背包问题

例题:装箱问题 ( http://www.wikioi.com/problem/1014/  ) 题目描述 有一个箱子容量为V(正整数,0<=V<=20000),同时有n个物品(0<n<=30),每个物品有一个体积(正整数). 要求n个物品中,任取若干个装入箱内,使箱子的剩余空间为最小. 输入描述 一个整数v,表示箱子容量,一个整数n,表示有n个物品 接下来n个整数,分别表示这n 个物品的各自体积 输出描述 一个整数,表示箱子剩余空间. 样例输入 24 6 8 3 12 7 9 7

算法导论三剑客之 动态规划 0-1背包问题

1 #include "iostream" 2 using namespace std; 3 4 float MAX(float m1,float m2){ 5 if(m1>=m2) 6 return m1; 7 else 8 return m2; 9 } 10 11 float bag_Zero_One(int n,float v,float p[],float w[]){ 12 if(n==0||v==0) 13 return 0; 14 else{ 15 float m2;

动态规划——0-1背包问题

 0-1背包问题: 描述:给定n中物品和一背包.物品i的重量是wi,其价值为vi,背包的容量为c,问应如何选择装入背包中的物品,使得装入背包中的物品总价值最大? 0-1背包问题是一个特殊的整数规划问题. 设所给0-1背包问题的子问题; 其最优值为m(i,j),即m(i,j)是背包容量为j,可选择物品为i,i+1,-,n时0-1背包问题的最优值.由0-1背包问题的最优子结构性质,可以建立计算m(i,j)的递归式如下: NO1:递归实现 1 /* 2 *Description 递归实现 3 *设有一

动态规划01背包问题

动态规划0-1背包问题 ? 问题描写叙述: 给定n种物品和一背包.物品i的重量是wi,其价值为vi,背包的容量为C.问应怎样选择装入背包的物品,使得装 入背包中物品的总价值最大? ? 对于一种物品,要么装入背包,要么不装.所以对于一种物品的装入状态能够取0和1.我们设物品i的装入状态为xi,xi∈ (0,1),此问题称为0-11背包问题. 过程分析 数据:物品个数n=5,物品重量w[n]={0,2,2,6,5,4},物品价值V[n]={0,6,3,5,4,6}, (第0位,置为0,不參与计算,仅

动态规划初探 -- 背包问题

在为期一个星期的ACM集训之后,我就这样做了一个逃兵hhhh 在这一个星期里面,学长讲了快速排序,二分三分搜索,矩阵快速幂,线段树,BFS(广度优先搜索)和DFS(深度优先搜索),邻接表和哈希表,结构体和优先队列,背包问题和动态规划. 其中讲快速排序那天我还在考试,就没有去听,第二天找学长的时候也听得似懂非懂. 学长讲矩阵快速幂的时候爸妈来找我,也没有听.其中基本都学的不扎实. 所以都要后期重新再学一遍,巩固扎实.估计学长还会讲并查集和最小生成树,我就自己看看书吧~ ==============

【动态规划/多重背包问题】POJ1014-Dividing

多重背包问题的优化版来做,详见之前的动态规划读书笔记. dp[i][j]表示前i中数加得到j时第i种数最多剩余几个(不能加和得到i的情况下为-1)递推式为: dp[i][j]=mi(dp[i-1][j]≥0,即前i-1种数就能达到数字j) =-1(j<ai 或者 dp[i][j-ai]≤0,即再加上一个第i种数也无法达到j 或者 当前和小于当前数) =dp[i][j-ai]-1(可以达到的情况) #include<iostream> #include<cstdio> #inc

【算法数据结构Java实现】Java实现动态规划(背包问题)

1.背景 追随着buptwusuopu大神的脚步,最近在研习动态规划.动态规划应该叫一种解决问题的思想,记得又一次去某公司面试就被问到了这个. 多于动态规划的理解,大致是这样的,从空集合开始,每增加一个元素就求它的最优解,直到所有元素加进来,就得到了总的最优解. 比较典型的应用就是背包问题,有一个重量一定的包,有若干件物品,他们各自有不同的重量和价值,怎样背才能取得最大价值. 错误的理解:去价值/重量比最大的物品多装(之前我也是这么想的,但是在背包重量一定的情况下这么做并不合理,范例很容易想到)

动态规划0-1背包问题

最近看了一些简单的动态规划方面的例题 在学习的过程中发现 有的问题虽然不难 但是第一次看还是会有些问题 所以把自己弄0-1背包的问题拿出来给大家分享 不喜勿喷 网上资源特别多 讲解什么的就算了 其他人画的图都不错 递推关系: 设所给0-1背包问题的最优值为m(i,j),即m(i,j)是背包容量为j,可选择物品为i,i+1,-,n时0-1背包问题的最优值.由0-1背包问题的最优子结构性质,可以建立计算m(i,j)的递归式: <!--[endif]--> 上式此时背包容量为j,可选择物品为i.此时

动态规划小结——背包问题

背包问题是动态规划的经典问题,所有初次学习动态规划都会讲述这个经典问题.因此,有必要弄清跟背包问题的所有分析过程并熟练掌握各种类型的代码. 一,完全背包问题 1.问题描述:有n种物品,每种物品有无限多个,第i个物品重量是wi,价值是vi,从这些物品中挑选总重量不超过W的物品,求出挑选物品价值的最大值. 限制条件:1≤n≤100,1≤wi,vi≤100,1≤W≤10000 2.解题思路:本题类似于"硬币问题",硬币问题只要求凑够相应的面值,这里只不过多了一个新的属性--价值,即不仅要凑够