zoukankan      html  css  js  c++  java
  • 背包入门(01背包,完全背包,多重背包)

    推荐一篇好文章:http://blog.csdn.net/sj13051180/article/details/6687674

    0-1背包,每件物品只有一个,对于每件物品只有取与不取两种情况,

    完全背包,每件物品无数多个,对于每件物品可以取任意多个。

    多重背包,每件物品有限多个,对于每件物品可以取任意多个。

    以poj-3624与poj-1276为原型,阐释三种背包的代码。

    0-1背包:

    #include <iostream>
    #include <stdio.h>
    using namespace std;
    int main(){
        int dp[12900];
        int n,m;
        int v[3405];
        int w[3405];
        while(cin >> n >> m){
            for(int j=0;j<=m;j++)
                dp[j]=0;
            for(int i=1;i<=n;i++)
                cin >> w[i] >> v[i];
            for(int i=1;i<=n;i++)
                for(int j=m;j>=w[i];j--)
                    dp[j] = max(dp[j],dp[j-w[i]]+v[i]);
            for(int j=0;j<=m;j++){
                cout << dp[j] <<" ";
            }
            cout <<endl;
            cout << dp[m] <<endl;
        }
        return 0;
    }

    完全背包:

    #include <iostream>
    #include <stdio.h>
    using namespace std;
    int main(){
        int dp[12900];
        int n,m;
        int v[3405];
        int w[3405];
        int num[3405];
        while(cin >> n >> m){
            for(int j=0;j<=m;j++)
                dp[j]=0;
            for(int i=1;i<=n;i++)
                cin >> w[i] >> v[i] ;
            for(int i=1;i<=n;i++)
                for(int k=1;k<=m/w[i];k++)
                    for(int j=k*w[i];j<=m;j++)
                        dp[j]=max(dp[j],dp[j-k*w[i]]+k*v[i]);
            for(int j=0;j<=m;j++){
                cout << dp[j] <<" ";
            }
            cout <<endl;
            cout << dp[m] <<endl;
        }
        return 0;
    }

    多重背包:

    三重循环,二维数组暴力解法(超时,超空间)

    #include <iostream>
    #include <stdio.h>
    #include <string.h>
    using namespace std;
    int dp[1406][1090];
    int main(){
    
        int n,m;
        int v[3405];
        int w[3405];
        int num[3405];
        while(cin >> n >> m){
            memset(dp,0,sizeof(dp));
            for(int i=1;i<=n;i++)
                cin >> w[i] >> v[i] >>num[i];
            for(int i=1;i<=n;i++)
                for(int k=1;k<=num[i];k++)
                    for(int j=m;j>=k*w[i];j--)
                        dp[i][j]=max(dp[i-1][j],dp[i-1][j-k*w[i]]+k*v[i]);
            for(int i=0;i<=n;i++){
                for(int j=0;j<=m;j++){
                    cout << dp[i][j] << " ";
                }
                cout << endl;
            }
            cout << dp[n][m] <<endl;
        }
        return 0;
    }

    三重循环,一位数组(超时)

    实现时,用临时的一维数组暂存i-1层物品的价值。

    #include <string.h>
    #include <iostream>
    #include <stdio.h>
    using namespace std;
    int main(){
        int dp[12900];
        int temp[12900];
        int n,m;
        int v[3405];
        int w[3405];
        int num[3405];
        while(cin >> n >> m){
            memset(dp,0,sizeof(dp));
            memset(temp,0,sizeof(temp));
            for(int i=1;i<=n;i++)
                cin >> w[i] >> v[i] >>num[i];
            for(int i=1;i<=n;i++){
                for(int j=0;j<=m;j++){
                    cout << dp[j] <<" ";
                }
                cout << endl;
                for(int i=0;i<=m;i++)
                    temp[i]=dp[i];
                for(int k=1;k<=num[i];k++){
                    for(int j=m;j>=k*w[i];j--){
                        dp[j]=max(dp[j],temp[j-k*w[i]]+k*v[i]);
                    }
                }
            }
            for(int j=0;j<=m;j++){
                cout << dp[j] <<" ";
            }
            cout << endl;
            cout << dp[m] <<endl;
        }
        return 0;
    }
    /*
    3 5
    1 60 2
    2 100 2
    3 120 2
    
    0 0 0 0 0 0
    0 60 120 120 120 120
    0 60 120 160 220 260
    0 60 120 160 220 260
    260
    */

    双重循环,一位数组(省时省空间)

    二进制压缩,把多数量的每种物品划分成单数量的多种物品,再运用0-1背包求解。

    #include <iostream>
    #include <string.h>
    #include <stdio.h>
    using namespace std;
    int dp[100005];
    int w[1000];
    int main(){
        int n,m,a,b,cnt;
        while(cin >> n >> m){
            cnt=0;
            memset(dp,0,sizeof(dp));
            memset(w,0,sizeof(w));
            for(int i=1;i<=m;i++){
                cin >> a >> b;
                for(int j=1;j<=a;j*=2){
                    w[++cnt]=j*b;
                    a-=j;
                }
                if(a>0){
                    w[++cnt]=a*b;
                }
            }
            for(int i=1;i<=cnt;i++)
                for(int j=n;j>=w[i];j--)
                    dp[j]=max(dp[j],dp[j-w[i]]+w[i]);
            cout << dp[n] <<endl;
        }
        return 0;
    }
  • 相关阅读:
    带你了解什么是Push消息推送
    小白学习如何打日志
    最近学到的前后端分离知识
    记一次愚蠢的操作--线程安全问题
    在工作中常用到的SQL
    在工作中常用的Linux命令
    记一次愚蠢的经历--String不可变性
    在公司做的项目和自己在学校做的有什么区别?
    程序员的快乐就是这么朴素无华且枯燥
    《大型网站系统与Java中间件》读书笔记 (中)
  • 原文地址:https://www.cnblogs.com/markliu/p/2795921.html
Copyright © 2011-2022 走看看