[atARC096F]Sweet Alchemy

给定一棵有根树,记$f_{i}$表示$i$的父亲,每一个点有一个代价$c_{i}$

给定常数$D$和$X$,再给每个点赋一个权值$v_{i}$($v_{i}\ge 0$),满足以下条件下最大化$\sum_{i=1}^{n}v_{i}$

条件:1.$\forall 2\le i\le n,v_{f_{i}}\le v_{i}\le v_{f_{i}}+D$;2.$\sum_{i=1}^{n}c_{i}v_{i}\le X$

$2\le n\le 50$,$1\le f_{i}<i$,$0\le D\le 10^{9}$,$1\le X,c_{i}\le 10^{9}$

构造$d_{i}=v_{i}-v_{f_{i}}$(特别的$d_{1}=v_{1}$),那么$v_{i}=d_{i}+v_{f_{i}}=...=\sum_{j为i祖先}d_{j}$

考虑每一个点的贡献,则有$\sum_{i=1}^{n}v_{i}=\sum_{i=1}^{n}d_{i}sz_{i}$和$\sum_{i=1}^{n}c_{i}v_{i}=\sum_{i=1}^{n}d_{i}sum_{i}$(其中$sz_{i}$为以$i$为根的子树大小,$sum_{i}$为以$i$为根子树中所有$c_{i}$的和)

可以看作一个背包,$n$个物品质量为$sum_{i}$,价值为$sz_{i}$,数量限制为$D$

先将所有物品按照“性价比”$\frac{sz_{i}}{sum_{i}}$从大到小排序,则若存在$x<y$满足$d_{x}+sz_{y}\le D$且$d_{y}\ge sz_{x}$,那么不妨令$d'_{y}=d_{y}-sz_{x}$且$d'_{x}=d_{x}+sz_{y}$,显然质量之和不变且价值不减少,因此最优解中必然不存在这类点对

由于$sz_{i}\le n$,将条件收缩为存在$x<y$满足$d_{x}+n\le D$且$d_{y}\ge n$

进一步的,将最终答案中的$d_{i}$分为三类:1.$D-n<d_{i}\le D$;2.$n\le d_{i}\le D-n$;3.$0\le d_{i}<n$,根据上述结论,任意第$i$类一定要在任意第$i+1$类前,且第2类数量不超过1(形象地,即构成了111……2333……)

对于第1类,转化为$D-n+1+d'_{i}$;对于第2类,转化为$d'_{i}+\lfloor\frac{d_{i}}{n}\rfloor n$,则都有$0\le d'_{i}<n$

先对$d'_{i}$背包,由于价值范围为$o(n^{3})$,那么令$f[i][j]$表示前$i$个物品,价值和为$j$的最小质量来dp即可,时间复杂度为$o(n^{4}\ln n)$或$o(n^{4})$(前者利用物品数量小于$\frac{n^{3}}{sz_{i}}$,后者用单调队列

枚举$\sum_{i=1}^{n}d'_{i}sz_{i}$(共$o(n^{3})$种),根据第$i$类要在第$i+1$类前的性质,对于剩下的部分贪心即可

(过程一眼看可能会有点假,注意$d'_{i}$是由$d_{i}$强制构造出来的,然后贪心根据的是结论)

特别的,对于$sz_{x}=n$这个位置(即原来的根),其没有数量限制,以第2类的方式构造$d'_{i}$,然后其贪心时没有上限

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 #define N 105
 4 #define ll long long
 5 struct ji{
 6     int nex,to;
 7 }edge[N];
 8 int E,n,m,d,x,head[N],id[N],sz[N],f[2][N*N*N];
 9 ll ans,sum[N];
10 bool cmp(int x,int y){
11     return sz[x]*sum[y]>sz[y]*sum[x];
12 }
13 void add(int x,int y){
14     edge[E].nex=head[x];
15     edge[E].to=y;
16     head[x]=E++;
17 }
18 void dfs(int k){
19     sz[k]=1;
20     for(int i=head[k];i!=-1;i=edge[i].nex){
21         dfs(edge[i].to);
22         sz[k]+=sz[edge[i].to];
23         sum[k]+=sum[edge[i].to];
24     }
25 }
26 int main(){
27     scanf("%d%d%d%lld",&n,&m,&d,&sum[1]);
28     memset(head,-1,sizeof(head));
29     for(int i=2;i<=n;i++){
30         scanf("%lld%d",&sum[i],&x);
31         add(x,i);
32     }
33     dfs(1);
34     for(int i=1;i<=n;i++)id[i]=i;
35     sort(id+1,id+n+1,cmp);
36     int mm=n*n*n;
37     for(int i=1;i<=mm;i++)f[0][i]=m+1;
38     for(int i=1;i<=n;i++){
39         int p=(i&1);
40         for(int j=0;j<=mm;j++){
41             f[p][j]=m+1;
42             for(int k=0;(k<=min(n,d))&&(k*sz[id[i]]<=j);k++)
43                 f[p][j]=min((ll)f[p][j],f[p^1][j-k*sz[id[i]]]+k*sum[id[i]]);
44         }
45     }
46     for(int i=0;i<=mm;i++)
47         if (f[(n&1)][i]<=m){
48             int s=m-f[(n&1)][i];
49             ll tot=i;
50             for(int j=1;j<=n;j++)
51                 if ((id[j]==1)||(sum[id[j]]*(d-n)>s)){
52                     tot+=1LL*sz[id[j]]*(s/sum[id[j]]);
53                     s-=sum[id[j]]*(s/sum[id[j]]);
54                 }
55                 else{
56                     if (d<n)continue;
57                     tot+=1LL*sz[id[j]]*(d-n);
58                     s-=sum[id[j]]*(d-n);
59                 }
60             ans=max(ans,tot);
61         }
62     printf("%lld",ans);
63 }
View Code

 

posted @ 2020-11-14 15:20  PYWBKTDA  阅读(132)  评论(0编辑  收藏  举报