zoukankan      html  css  js  c++  java
  • 动态规划之0-1背包问题

          给定n个物品和一个背包,物品i的重量是wi,其价值是vi,背包的容量为w,及最大载重量不超过W,在限定的总重量W内,我们如何选择物品,才能使物品的总价值最大。

      具体问题:

          有编号分别为a,b,c,d,e的五件物品,它们的重量分别是2,2,6,5,4,它们的价值分别是6,3,5,4,6,现在给你个承重为10的背包,如何让背包里装入的物品具有最大的价值总和?

         m(i,j)表示当前背包的容量为j,可选择的物品范围为i,i+1,i+2....n。所以背包问题的递推式如下

         m(i,j)={max{ m(i+1,j) ,m(i+1,j-w[i])+v[i]}(wi<=j), m(i+1,j)(wi>j)};(但看公式还是比较抽象的)

    下面具体的分析

    现在我们才去的分析方法是从左往右,从下往上。

    首先动态规划过程的表如下


    这张表是从下往上从左往右的分析

    当i=5时表示当前只有e一个物体,j从1到10慢慢增加表示背包容量在不断的增大,当j<=3的时候,背包的当前容量是小于e的,所以此时不能将物品加入到背包当中。当j>=4的时候恰好只有一个e所以背包最大为6。

    当i=1,j=8的时候m[2][8]=9(不加上a),当加上a之后之后m[i+1][j-w[i]]+v[i]=m[2][6]+v[1]=9+6=15.根据前面的公式m[i][j]=m[1][8]=15/其他的分析也是一样的。不过要注意顺序是从左往右,从下往上。

    代码如下

    #include <iostream>
    
    using namespace std;
    
    int n=5;//物品的件数
    int c=10;//背包的容量
    int w[]={0,2,2,6,5,4};//物品的重量
    int v[]={0,6,3,5,4,6};//物品的价值
    int m[6][11]={0};//记录规划过程
    int x[100];//保存路径
    
    void knapsack()
    {
        for(int i=0;i<=c;i++)
            if(w[n]<i)
                m[n][i]=v[n];
            else
            m[n][i]=0;
    
        //放置剩余的n-1个元素
        int i;
        for(i=n-1;i>=1;i--)
            for(int j=0;j<=c;j++)
                if(w[i]>j)
                    m[i][j]=m[i+1][j];
                else
                    m[i][j]=m[i+1][j]>m[i+1][j-w[i]]+v[i]?m[i+1][j]:m[i+1][j-w[i]]+v[i];
    }
    
    void tracesack()
    {
        int t=c;
        for(int i=1;i<n;i++)
            if(m[i][t]==m[i+1][t])
                x[i]=0;
            else
            {
                x[i]=1;
                t-=w[i];
            }
            x[n]=m[n][t]?1:0;
    }
    
    int main()
    {
        knapsack();
        tracesack();
        for(int i=1;i<=n;i++)
            printf("%d ",x[i]);
        printf("
    %d",m[1][c]);
        return 0;
    }
    


  • 相关阅读:
    归并排序,树状数组 两种方法求逆序对
    volley源代码解析(六)--HurlStack与HttpClientStack之争
    what&#39;s new in vc2015
    [ajax 学习笔记] ajax初试
    安卓项目开发实战(1)--首页顶部菜单BAR实现
    eclipse下Tomcat7.0启动奔溃问题
    伸缩--也可用于tabs
    怎样提高团队管理能力8
    mysql查询今天,昨天,近7天,近30天,本月,上一月数据
    SDSoC使用体验
  • 原文地址:https://www.cnblogs.com/gaot/p/7709714.html
Copyright © 2011-2022 走看看