题面传送门

题意:

  • 给你一棵 \(n\) 个顶点的树和 \(m\) 条带权值的附加边
  • 你要选择一些附加边加入原树中使其成为一个仙人掌(每个点最多属于 \(1\) 个简单环)
  • 求你选择的附加边权值之和的最大值。
  • \(n \in [3,2 \times 10^5]\),\(m \in [0,2 \times 10^5]\)

很容易发现,如果你选择一条端点为 \(u,v\) 的附加边,那么 \(u\) 到 \(v\) 路径上所有点都会被包含进一个简单环。

那么题目变为:有 \(m\) 条路径,你要选择其中某些路径使得每个点最多被一条路径覆盖。问选择的路径权值之和的最大值。

对于这一类问题,我们可以想到树形 \(dp\)。记 \(dp_u\) 为:选择一些路径,路径上所有点都在 \(u\) 的子树内,并且路径两两不相交,这些路径权值之和的最大值。

转移分两种情况:

  1. \(u\) 没有被覆盖,那么 \(dp_u=\sum\limits_{fa_v=u}dp_v\)
  2. \(u\) 被覆盖。那么显然它只能被 \(LCA=u\) 的路径覆盖(因为根据之前的定义,路径上所有节点都必须在 \(u\) 的子树内)。这种情况下就是 \(u\) 的子树挖掉这条路径后剩余部分的答案+这条路径的权值。

    是不是听起来有些抽象?不妨画个图理解看:

    假设有如下图的树,你选择了由橙色边组成的路径:



    我们把这条路径上的点全部去掉看看剩余部分是个什么东西。



    凭直觉,这部分的贡献为 \(dp[4]+dp[5]+dp[6]+dp[7]+dp[10]+dp[13]+dp[16]+dp[17]+dp[19]+dp[21]+dp[24]\)

    发现了什么了吗?

    你可能会说,项数这么多,我啥也没发现。

    那我们就把删掉的点补全后看看呗。



    通过观察,我们发现:
  • \(4,5\) 是 \(3\) 的儿子
  • \(6,7\) 是 \(2\) 除了 \(3\) 之外的儿子
  • \(10,13\) 是 \(1\) 除了 \(2,15\) 之外的儿子
  • \(16,17\) 是 \(15\) 除了 \(18\) 之外的儿子
  • \(19\) 是 \(18\) 除了 \(20\) 之外的儿子
  • \(21,24\) 是 \(20\) 的儿子。

    我们记 \(p_1 \to p_2 \to \dots \to p_k\) 为我们选择的路径,其中 \(p_j=i\),\(p_0=p_{k+1}=0\),那么贡献为:
\[\sum\limits_{i=1}^j\sum\limits_{fa_v=p_i}dp_v \times [v \neq p_{i-1}]+\sum\limits_{fa_v=u}dp_v \times [v \neq p_{i-1} \and v \neq p_{i+1}]+\sum\limits_{i=j+1}^k\sum\limits_{fa_v=p_i}dp_v \times [v \neq p_{i+1}]+w
\]

暴力显然是 \(n^2\) 的,考虑对其进行优化。

记 \(g_x=\sum\limits_{fa_y=x}dp_y-dp_x\),那么上式子等价于

\[\sum\limits_{i=1}^{j-1}g_{p_i}+\sum\limits_{fa_v=u}dp_v+\sum\limits_{i=j+1}^kg_{p_i}+w
\]

用树链剖分随便维护一下就可以了

/*
Contest: -
Problem: Codeforces 856D
Author: tzc_wk
Time: 2020.8.20
*/
#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define pb push_back
#define fz(i,a,b) for(int i=a;i<=b;i++)
#define fd(i,a,b) for(int i=a;i>=b;i--)
#define foreach(it,v) for(__typeof(v.begin()) it=v.begin();it!=v.end();it++)
#define all(a) a.begin(),a.end()
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a))
#define fillbig(a) memset(a,0x3f,sizeof(a))
#define y1 y1010101010101
#define y0 y0101010101010
typedef pair<int,int> pii;
typedef long long ll;
inline int read(){
int x=0,neg=1;char c=getchar();
while(!isdigit(c)){
if(c=='-') neg=-1;
c=getchar();
}
while(isdigit(c)) x=x*10+c-'0',c=getchar();
return x*neg;
}
int n=read(),m=read();
int head[200005],to[400005],nxt[400005],ecnt=0;
inline void adde(int u,int v){
to[++ecnt]=v;nxt[ecnt]=head[u];head[u]=ecnt;
}
int fa[200005],top[200005],dep[200005],siz[200005],wson[200005],dfn[200005],nd[200005],tim=0;
inline void dfs1(int x){
siz[x]=1;
for(int i=head[x];i;i=nxt[i]){
int y=to[i];dep[y]=dep[x]+1;dfs1(y);siz[x]+=siz[y];
if(siz[y]>siz[wson[x]]) wson[x]=y;
}
}
inline void dfs2(int x,int tp){
top[x]=tp;dfn[x]=++tim;nd[tim]=x;
if(wson[x]) dfs2(wson[x],tp);
for(int i=head[x];i;i=nxt[i]){
int y=to[i];if(y!=wson[x]) dfs2(y,y);
}
}
inline int getlca(int x,int y){
while(top[x]!=top[y]){
if(dep[top[x]]<dep[top[y]]) swap(x,y);
x=fa[top[x]];
}
if(dep[x]<dep[y]) swap(x,y);
return y;
}
struct path{
int u,v,w;
path(){/*nothing*/}
path(int _u,int _v,int _w){u=_u;v=_v;w=_w;}
}; vector<path> t[200005];
struct node{
int l,r,val;
} s[200005<<2];
inline void build(int k,int l,int r){
s[k].l=l;s[k].r=r;
if(l==r) return;
int mid=(l+r)>>1;
build(k<<1,l,mid);
build(k<<1|1,mid+1,r);
}
inline void modify(int k,int ind,int val){
if(s[k].l==s[k].r){s[k].val+=val;return;}
int mid=(s[k].l+s[k].r)>>1;
if(ind<=mid) modify(k<<1,ind,val);
else modify(k<<1|1,ind,val);
s[k].val=s[k<<1].val+s[k<<1|1].val;
}
inline int query(int k,int l,int r){
if(l<=s[k].l&&s[k].r<=r) return s[k].val;
int mid=(s[k].l+s[k].r)>>1;
if(r<=mid) return query(k<<1,l,r);
else if(l>mid) return query(k<<1|1,l,r);
else return query(k<<1,l,mid)+query(k<<1|1,mid+1,r);
}
inline int calc(int u,int v){
int sum=0;
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]]) swap(u,v);
sum+=query(1,dfn[top[u]],dfn[u]);u=fa[top[u]];
}
if(dep[u]<dep[v]) swap(u,v);
sum+=query(1,dfn[v],dfn[u]);return sum;
}
int dp[200005];
inline void dfs3(int x){
int sum=0;
for(int i=head[x];i;i=nxt[i]){
dfs3(to[i]);sum+=dp[to[i]];
}
dp[x]=sum;
foreach(it,t[x]) dp[x]=max(dp[x],calc(it->u,it->v)+it->w+sum);
modify(1,dfn[x],sum-dp[x]);
}
int main(){
for(int i=2;i<=n;i++){fa[i]=read();adde(fa[i],i);}
dfs1(1);dfs2(1,1);build(1,1,n);
for(int i=1;i<=m;i++){
int u=read(),v=read(),w=read();
t[getlca(u,v)].pb(path(u,v,w));
}
dfs3(1);printf("%d\n",dp[1]);
return 0;
}

Codeforces 856D - Masha and Cactus(树链剖分优化 dp)的更多相关文章

  1. 【BZOJ4712】洪水 树链剖分优化DP+线段树

    [BZOJ4712]洪水 Description 小A走到一个山脚下,准备给自己造一个小屋.这时候,小A的朋友(op,又叫管理员)打开了创造模式,然后飞到山顶放了格水.于是小A面前出现了一个瀑布.作为 ...

  2. 【CF1009F】Dominant Indices(长链剖分优化DP)

    点此看题面 大致题意: 设\(d(x,y)\)表示\(x\)子树内到\(x\)距离为\(y\)的点的个数,对于每个\(x\),求满足\(d(x,y)\)最大的最小的\(y\). 暴力\(DP\) 首先 ...

  3. CF1009F Dominant Indices——长链剖分优化DP

    原题链接 \(EDU\)出一道长链剖分优化\(dp\)裸题? 简化版题意 问你每个点的子树中与它距离为多少的点的数量最多,如果有多解,最小化距离 思路 方法1. 用\(dsu\ on\ tree\)做 ...

  4. CodeForces 916E Jamie and Tree(树链剖分+LCA)

    To your surprise, Jamie is the final boss! Ehehehe. Jamie has given you a tree with n vertices, numb ...

  5. (中等) HDU 5293 Tree chain problem,树链剖分+树形DP。

    Problem Description   Coco has a tree, whose vertices are conveniently labeled by 1,2,…,n.There are ...

  6. [NOIP2018提高组] 保卫王国 (树链剖分+动态DP)

    题面 题目链接-Luogu 题目链接-Loj(要加Freopen) 题解 什么是动态DP? OneInDark:你不需要知道这么多,你只需要知道是利用了广义矩阵乘法就够了! 广义矩乘 广义矩阵乘法,简 ...

  7. 2019.01.19 bzoj3653: 谈笑风生(长链剖分优化dp)

    传送门 长链剖分优化dpdpdp水题. 题意简述:给一棵树,mmm次询问,每次给一个点aaa和一个值kkk,询问满足如下条件的三元组(a,b,c)(a,b,c)(a,b,c)的个数. a,b是c的祖先 ...

  8. 长链剖分优化dp三例题

    首先,重链剖分我们有所认识,在dsu on tree和数据结构维护链时我们都用过他的性质. 在这里,我们要介绍一种新的剖分方式,我们求出这个点到子树中的最长链长,这个链长最终从哪个儿子更新而来,那个儿 ...

  9. Codeforces 909C Python Indentation:树状数组优化dp

    题目链接:http://codeforces.com/contest/909/problem/C 题意: Python是没有大括号来标明语句块的,而是用严格的缩进来体现. 现在有一种简化版的Pytho ...

随机推荐

  1. javascript-jquery插件

    1.jquery创建插件 jQuery.extend({插件名:函数体,插件名:函数体}): html部分 <div id="div1">开始动画</div> ...

  2. OO第四单元及学期总结

    OO第四单元及学期总结 第四单元两次作业的架构设计 第一次作业 类图: 树形结构:使用Operation类管理UMLOperation以及parent为该UMLOperation的参数(UMLpara ...

  3. SpringCloud微服务实战——搭建企业级开发框架(八):使用注解校验微服务消息参数

      平时开发过程中,经常要用到参数校验,如果直接在代码逻辑里面写参数校验,代码有点冗余且用起来不是非常方便,显得代码逻辑复杂且重复代码太多,这里我们使用注解的方式进行参数校验,SpringBoot中常 ...

  4. [HNOI2009]双递增序列(洛谷P4728)+小烈送菜(内部训练题)——奇妙的dp

    博主学习本题的经过嘤嘤嘤: 7.22 : 听学长讲(一知半解)--自己推(推不出来)--网上看题解--以为自己会了(网上题解是错的)--发现错误以后又自己推(没推出来)--给学长发邮件--得到正确解法 ...

  5. Machine learning(2-Linear regression with one variable )

    1.Model representation Our Training Set [训练集]: We will start with this ''Housing price prediction'' ...

  6. F. Mattress Run 题解

    F. Mattress Run 挺好的一道题,对于DP的本质的理解有很大的帮助. 首先要想到的就是将这个拆成两个题,一个dp光求获得足够的夜晚的最小代价,一个dp光求获得足够的停留的最小代价. 显然由 ...

  7. PHP笔记3__简易计算器

    <?php header("Content-type: text/html; charset=utf-8"); error_reporting(E_ALL & ~E_ ...

  8. 第07课 OpenGL 光照和键盘(2)

    下一段代码绘制贴图立方体.我只对新增的代码进行注解.如果您对没有注解的代码有疑问,回头看看第六课. int DrawGLScene(GLvoid) // 从这里开始进行所有的绘制 { glClear( ...

  9. html5的Message信息提示框

    1.定义dialog <dialog id="showDialog"> <div>您没有权限访问本页面!</div> </dialog&g ...

  10. upload-labs通关攻略(全)

    upload-labs通关攻略 upload-labs是练习文件上传很好的一个靶场,建议把upload-labs关卡全部练习一遍 1.下载安装 下载地址 链接:https://pan.baidu.co ...