zoukankan      html  css  js  c++  java
  • 01背包问题之2(dp)

    01背包问题之2

    有n个物品,重量和价值分别为wi和vi,从这些物品中挑选出重量不超过W的物品,求所有挑选方案中物品价值总和的最大值

    限制条件:

      1 <= n <= 100; 1 <= wi<= 10^7; 1 <= vi <= 100; 1 <= W <= 10^9;

    分析:数据量更大,之前求解该问题的时间复杂度为o(nW),在这一问题来说会超时,在这个问题里重量很大,但是价值很小,可以考虑价值,改变dp的对象,针对不同的价值来计算最小的质量

       因为是求最小值,开始把dp初始化为INF

    状态:dp[i][j] = 前i个物品中挑选价值总和为j时的重量最小值

    状态转移方程:dp[i+1][j] = min(dp[i][j], dp[i][j - v[i]] + w[i]);

    利用翻滚数组即一维数组可以大大节省空间

    代码:

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    using namespace std;
    const int INF = 1000000001; 
    int dp[10005];
    int w[105], v[105];
    int main() {
        int n, W;
        while(scanf("%d%d", &n, &W) == 2) {
            int sum = 0;
            for(int i = 0; i < n; i++) {
                scanf("%d%d", &w[i], &v[i]);
                sum += v[i];    
            }
            fill(dp, dp + 10005, INF);
            dp[0] = 0;
            for(int i = 0; i < n; i++) {
                for(int j = sum; j >= v[i]; j--) {
                     dp[j] = min(dp[j], dp[j - v[i]] + w[i]);
                } 
            }
            for(int i = sum; i >= 0; i--) {
                if(dp[i] <= W) {
                    printf("%d
    ", i); 
                    break;    
                }
            }
        }
        return 0;
    }
    作者:kindleheart
    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须在文章页面给出原文连接,否则保留追究法律责任的权利。
  • 相关阅读:
    金融新手投标模块布局小Demo
    jQuery序列化Ajax提交表单
    javascript实现jsonp跨域问题+原理
    javascript返回顶部插件+源码
    mime中间件
    移动端meta标签的设置
    Node环境下实现less编译
    diogo谈框,仿prompt()方法布局
    linux驱动程序框架基础
    C/C++下Netbeans的配置
  • 原文地址:https://www.cnblogs.com/kindleheart/p/8870521.html
Copyright © 2011-2022 走看看