CSUST 黄金矿工 题解(分组背包+转换dp方程状态)

题目链接

题目思路

算是两个经典问题的结合

首先看到问题描述可以转换为分组背包

看到\(t\)很大,所以设\(dp[i]\)表示达到价值为\(i\)的最少时间是多少,然后\(dp\)即可

代码

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
#define fi first
#define se second
#define debug printf("aaaaaaaaaaa\n");
const int maxn=2e2+100,inf=0x3f3f3f3f,mod=1e9+7;
const ll INF=0x3f3f3f3f3f3f3f3f;
const double eps=1e-7;
int n,sum;
int v[maxn],t[maxn],f[maxn];
int son[maxn],fa[maxn];
vector<pii> vec[maxn];
ll dp[10000+5];
int main(){
    int _; scanf("%d",&_);
    while(_--){
        scanf("%d%d",&n,&sum);
        for(int i=1;i<=n;i++){
            fa[i]=son[i]=0;
            vec[i].clear();
        }
        for(int i=1;i<=n;i++){
            scanf("%d%d%d",&v[i],&t[i],&f[i]);
            son[f[i]]=i;
            fa[i]=f[i];
        }
        int cnt=0;
        for(int i=1;i<=n;i++){
            if(fa[i]!=0) continue;
            cnt++;
            int x=i;
            int tempv=0,tempt=0;
            while(x){
                tempv+=v[x];
                tempt+=t[x];
                vec[cnt].push_back({tempv,tempt});
                x=son[x];
            }
        }
        memset(dp,0x3f,sizeof(dp));
        dp[0]=0;
        for(int i=1;i<=cnt;i++){
            for(int j=10000;j>=0;j--){
                for(int k=0;k<vec[i].size();k++){
                    int x=vec[i][k].fi,y=vec[i][k].se;
                    if(j>=x){
                        dp[j]=min(dp[j],dp[j-x]+y);
                    }
                }
            }
        }
        for(int i=10000;i>=0;i--){
            if(dp[i]<=sum){
                printf("%d\n",i);
                break;
            }
        }
    }
    return 0;
}


posted @ 2021-08-03 20:20  hunxuewangzi  阅读(56)  评论(0编辑  收藏  举报