2022 icpc杭州站 C. No Bug No Game - 背包dp

题面

分析

能拿整个 p i p_i pi的就拿整个的,不能拿了可以拿一部分的,因此可以分成0和1两种情况,0表示拿整个的,1表示还可以拿部分的,两种情况放在一起做一遍01背包,找到最大价值。

代码
cpp 复制代码
#include <bits/stdc++.h>

#define int long long

using namespace std;

const int N = 3010;

int dp[N][N][2];
int w[N][20];
int p[N];

signed main() {
    ios::sync_with_stdio(false);
    cin.tie(nullptr);

    int n, m;
    cin >> n >> m;
    int sum = 0;
    for(int i = 1; i <= n; i ++) {
        cin >> p[i];
        sum += p[i];
        for(int j = 1; j <= p[i]; j ++) cin >> w[i][j];
    }
    if(sum <= m) {
        int ans = 0;
        for(int i = 1; i <= n; i ++) ans += w[i][p[i]];
        cout << ans << "\n";
        return 0;
    }
    memset(dp, -0x3f, sizeof dp);
    dp[0][0][0] = 0;
    for(int i = 1; i <= n; i ++) {
        for(int j = 0; j <= m; j ++) {
            dp[i][j][0] = dp[i - 1][j][0];
            dp[i][j][1] = dp[i - 1][j][1];
            if(j >= p[i]) {
                dp[i][j][0] = max(dp[i][j][0], dp[i - 1][j - p[i]][0] + w[i][p[i]]);
                dp[i][j][1] = max(dp[i][j][1], dp[i - 1][j - p[i]][1] + w[i][p[i]]);
            }
            for(int k = 1; k < p[i]; k ++) {
                if(j >= k) dp[i][j][1] = max(dp[i][j][1], dp[i - 1][j - k][0] + w[i][k]);
            }
        }
    }
    cout << max(dp[n][m][0], dp[n][m][1]) << "\n";
}
相关推荐
记得早睡~21 分钟前
leetcode51-N皇后
javascript·算法·leetcode·typescript
Zhichao_971 小时前
【UE5 C++课程系列笔记】32——读Json文件并解析
c++·ue5
lancyu1 小时前
C语言--插入排序
c语言·算法·排序算法
点云SLAM1 小时前
C++20新增内容
c++·算法·c++20·c++ 标准库
照书抄代码1 小时前
C++11可变参数模板单例模式
开发语言·c++·单例模式·c++11
No0d1es1 小时前
CCF GESP C++编程 四级认证真题 2025年3月
开发语言·c++·青少年编程·gesp·ccf·四级·202503
No0d1es2 小时前
CCF GESP C++编程 五级认证真题 2025年3月
开发语言·c++·青少年编程·gesp·ccf·五级·2025年3月
shuaixio2 小时前
【C++代码整洁之道】第九章 设计模式和习惯用法
c++·设计模式·设计原则·常见设计模式·习惯用法
独好紫罗兰2 小时前
洛谷题单3-P1217 [USACO1.5] 回文质数 Prime Palindromes-python-流程图重构
开发语言·python·算法
独好紫罗兰2 小时前
洛谷题单2-P1424 小鱼的航程(改进版)-python-流程图重构
开发语言·python·算法