• 【01背包】Codeforces 730J


    题目链接

    题意

    现在有 n 瓶已经打开的可乐,每瓶可乐有两个值 \(a_i\):剩余的可乐容量,\(b_i\):瓶子的体积。

    现在可以把一个瓶子里的可乐倒向其他瓶子,倒体积为 \(x\) 的可乐花费的时间为 \(x\)

    问最少需要几个瓶子可以装这些可乐,以及倒可乐花费的最少时间。

    思路

    01背包

    首先直接按照可乐瓶子的体积排序,从大到小遍历可以求出最少瓶子数量 \(num\)

    因为倒可乐的花费等于倒的体积,所以我们选择的 \(num\) 个瓶子中可乐的剩余体积要尽可能的大,并且要保证总体积大于剩余可乐的总体积。

    其实就是加一维的01背包。

    \(dp[i][j][k]\) 表示在前 \(i\) 瓶可乐中,选择 \(j\) 瓶,总容量为 \(k\) 的可乐所剩余的可乐体积的最大值。

    优化掉第一维,转移方程如下:

    for (int i = 1; i <= n; i++) {
        for (int j = min(i, num); j; j--) {
            for (int k = sum2; k >= arr[i].v; k--) {
                dp[j][k] = max(dp[j][k], dp[j - 1][k - arr[i].v] + arr[i].w);
            }
        }
    }
    

    所需可乐瓶数量最少为 \(num\),剩余可乐总体积为 \(sum1\),所有瓶子的总体积 \(sum2\)

    我们遍历 \(dp[n][num][sum1 , sum2]\) 求最大值 \(maxn\)

    最后的花费即 \(sum1 -maxn\)

    感觉复杂度很高,但是循环常数比较小,可以写

    代码

    #include <algorithm>
    #include <iostream>
    #include <map>
    #include <math.h>
    #include <queue>
    #include <set>
    #include <stack>
    #include <stdio.h>
    #include <string.h>
    #include <string>
    #include <vector>
    #define emplace_back push_back
    #define pb push_back
    using namespace std;
    typedef long long ll;
    typedef unsigned long long ull;
    const int mod = 1e9 + 7;
    const int seed = 12289;
    const double eps = 1e-6;
    const int inf = 0x3f3f3f3f;
    const int N = 1e2 + 10;
    
    struct note {
        int w, v;
    } arr[N];
    bool cmp(note a, note b)
    {
        return a.v > b.v;
    }
    
    int dp[101][10001];
    
    int main()
    {
        int n;
        scanf("%d", &n);
        int sum = 0, sum2 = 0;
        for (int i = 1; i <= n; i++) {
            scanf("%d", &arr[i].w);
            sum += arr[i].w;
        }
        int rel = sum;
        for (int i = 1; i <= n; i++) {
            scanf("%d", &arr[i].v);
            sum2 += arr[i].v;
        }
        sort(arr + 1, arr + 1 + n, cmp);
        int num = 0;
        for (int i = 1; i <= n; i++) {
            sum -= arr[i].v;
            if (sum <= 0) {
                num = i;
                break;
            }
        }
        memset(dp, 0x8f, sizeof(dp));
        dp[0][0] = 0;
        for (int i = 1; i <= n; i++) {
            for (int j = min(i, num); j; j--) {
                for (int k = sum2; k >= arr[i].v; k--) {
                    dp[j][k] = max(dp[j][k], dp[j - 1][k - arr[i].v] + arr[i].w);
                }
            }
        }
        int ans = 0;
        for (int i = rel; i <= sum2; i++) {
            ans = max(ans, dp[num][i]);
        }
        printf("%d %d\n", num, rel - ans);
        return 0;
    }
    
  • 相关阅读:
    2.18周四笔记
    Tkinter 控件详细介绍
    python基础: day4作业计算器
    Python的functools.reduce用法
    正则表达式技术深入
    递归
    生成器generator
    绕过验证码
    导入JSONPathExtractorExample.jmx文件报错,导不进去
    jmeter一些插件下载网址
  • 原文地址:https://www.cnblogs.com/valk3/p/13966441.html
Copyright © 2020-2023  润新知