【题解】魔法树 Magic Tree [CEOI2019] [CF1193B] [Loj3166]

【题解】魔法树 Magic Tree [CEOI2019] [CF1193B] [Loj3166]

传送门:魔法树 \(\text{Magic Tree}\) \(\text{[CEOI2019]}\) \(\text{[CF1193B]}\) \(\text{[Loj3166]}\)

【题目描述】

给出一颗 \(n\) 个节点的树(根为 \(1\)),每个节点 \(i\) 会在第 \(d_i\) 天时放出量为 \(w_i\) 的妹汁。在每一天 \(j\) 您可以砍掉树上的任意一些边,若当天与节点 \(1\) 不再联通的点的 \(d_i\) 恰好等于 \(j\),您可以获得这些妹汁。

求总共最多能收获多少妹汁。

【分析】

先考虑暴力 \(dp\),设 \(f(x,j)\) 表示在时间 \(j\) 时以 \(x\) 为根的子树可获得的最大妹汁量,则有:

\(f(x,j)=\max\{f(x,j-1),g(x,j)\}\),其中 \(g(x,j)=w(x)*[j=d(x)]+\sum_{to\in son(x)}f(to,j)\)

复杂度 \(O(nk)\),可以过 \(\text{Subtask 1,4,5}\),有 \(34pts\)

用个小 \(trick\) 优化转移。

注意到 \(f\) 实际上是 \(g\) 的前缀最大值,给每个点开个 \(map\) 记录 \(j\)\(f(x,j)-f(x,k)\) \((k<j,f(x,k)\neq f(x,j))\)(相邻两个不同前缀最大值的差),那坨子树累加直接启发式合并。

每次给 \(g(x,d(x))\) 加上 \(w(x)\) 后都会使从 \(d(x)\) 开始的连续一段发生改变,先 \(\text{upperbound}\) 找到 \(d(x)\) 位置后面的第一个指针,然后依次往后移动并删去被更新掉的位置,无法继续更新时就停止(利用差分的良好性质)。

【Code】

#include<algorithm>
#include<cstring>
#include<cstdio>
#include<map>
#define LL long long
#define Re register int
#define IT map<int,LL>::iterator
using namespace std;
const int N=1e5+3;
int n,m,x,y,o,T,K,t[N],v[N],head[N];
struct QAQ{int to,next;}a[N];
inline void add(Re x,Re y){a[++o].to=y,a[o].next=head[x],head[x]=o;}
inline void in(Re &x){
    int f=0;x=0;char c=getchar();
    while(c<'0'||c>'9')f|=c=='-',c=getchar();
    while(c>='0'&&c<='9')x=(x<<1)+(x<<3)+(c^48),c=getchar();
    x=f?-x:x;
}
struct Sakura1{//暴力
    LL ans,dp[N][23];
    inline void dfs(Re x){
        for(Re i=head[x],to;i;i=a[i].next){
            dfs(to=a[i].to);
            for(Re j=0;j<=K;++j)dp[x][j]+=dp[to][j];
        }
        dp[x][t[x]]+=v[x];
        for(Re j=t[x]+1;j<=K;++j)dp[x][j]=max(dp[x][j],dp[x][j-1]);
    }
    inline void sakura(){dfs(1),printf("%lld\n",dp[1][K]);}
}T1;
struct Sakura2{//正解
    int id[N];LL ans;map<int,LL>s[N];
    inline void dfs(Re x){
        id[x]=x;
        for(Re i=head[x],to;i;i=a[i].next){
            dfs(to=a[i].to);
            if(s[id[x]].size()<s[id[to]].size())swap(id[x],id[to]);
            for(IT it=s[id[to]].begin();it!=s[id[to]].end();++it)
                s[id[x]][it->first]+=it->second;
        }
        s[id[x]][t[x]]+=v[x];LL dlt=v[x];IT it=s[id[x]].upper_bound(t[x]),tmp;
        while(it!=s[id[x]].end())
            if(dlt>=it->second)dlt-=it->second,tmp=it++,s[id[x]].erase(tmp);//f(x,d(x))更优,删掉这个点
            else{it->second-=dlt;break;}//这个点更优,结束枚举
    }
    inline void sakura(){
        dfs(1);
        for(IT it=s[id[1]].begin();it!=s[id[1]].end();++it)ans+=it->second;//差分回收
        printf("%lld\n",ans);
    }
}T2;
int main(){
//    freopen("magic.in","r",stdin);
//    freopen("magic.out","w",stdout);
    in(n),in(T),in(K),m=n-1;
    for(Re i=2;i<=n;++i)in(x),add(x,i);
    while(T--)in(x),in(t[x]),in(v[x]);
//    if(K<=20)T1.sakura();else
    T2.sakura();
}
posted @ 2020-05-21 16:27  辰星凌  阅读(498)  评论(0编辑  收藏  举报