[HDU 5293]Tree chain problem(树形dp+树链剖分)

时间:2022-09-07 17:05:53

[HDU 5293]Tree chain problem(树形dp+树链剖分)

题面

在一棵树中,给出若干条链和链的权值,求选取不相交的链使得权值和最大。

分析

考虑树形dp,dp[x]表示以x为子树的最大权值和(选的链都在i的子树中)

设sum[x]表示x的儿子的dp值和,即\(\sum _{y \in \mathrm{son}(x)} dp[y]\)

1.不选两端点lca为x的链,dp[x]=sum[x]

2.选两端点lca为x的链,则dp[x]=max{链的权值+链上节点的所有子节点dp的和 - 链上节点dp和}(因为x的子节点也可能在链上,要去掉),这样选的链一定和lca为x的链不相交。取max即可得到答案。如果不理解,请看下图。

[HDU 5293]Tree chain problem(树形dp+树链剖分)

路径为4->7,如果直接对sum求和,我们会把2,3,4,5,6,7都算进答案,但实际上我们只需要5,6,所以再做一次减法,把4->7路径上的点减掉就可以了

预处理的时候可以保存两端点lca为x的链的编号。看到路径求和,我们直接用树链剖分维护即可。从下到上dp,一边更新dp值和sum值,一边求和。

时间复杂度\(O(n\log^2n)\)

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#define maxn 100000
#define maxm 100000
using namespace std;
int t,n,m;
struct edge{
int from;
int to;
int next;
}E[maxn*2+5];
int head[maxn+5];
int esz=1;
void add_edge(int u,int v){
esz++;
E[esz].from=u;
E[esz].to=v;
E[esz].next=head[u];
head[u]=esz;
} struct segment_tree{
struct node{
int l;
int r;
long long sum;
}tree[maxn*4+5];
void push_up(int pos){
tree[pos].sum=tree[pos<<1].sum+tree[pos<<1|1].sum;
}
void build(int l,int r,int pos){
tree[pos].l=l;
tree[pos].r=r;
tree[pos].sum=0;
if(l==r) return;
int mid=(l+r)>>1;
build(l,mid,pos<<1);
build(mid+1,r,pos<<1|1);
push_up(pos);
}
void update(int upos,long long uval,int pos){
if(tree[pos].l==tree[pos].r){
tree[pos].sum+=uval;
return;
}
int mid=(tree[pos].l+tree[pos].r)>>1;
if(upos<=mid) update(upos,uval,pos<<1);
else update(upos,uval,pos<<1|1);
push_up(pos);
}
long long query(int L,int R,int pos){
if(L<=tree[pos].l&&R>=tree[pos].r){
return tree[pos].sum;
}
int mid=(tree[pos].l+tree[pos].r)>>1;
long long ans=0;
if(L<=mid) ans+=query(L,R,pos<<1);
if(R>mid) ans+=query(L,R,pos<<1|1);
return ans;
}
}T1,T2;//T1维护sum,T2维护dp
int tim;
int sz[maxn+5],deep[maxn+5],dfn[maxn+5],top[maxn+5],son[maxn+5],fa[maxn+5];
void dfs1(int x,int f){
fa[x]=f;
sz[x]=1;
deep[x]=deep[f]+1;
for(int i=head[x];i;i=E[i].next){
int y=E[i].to;
if(y!=f){
dfs1(y,x);
sz[x]+=sz[y];
if(sz[y]>sz[son[x]]) son[x]=y;
}
}
}
void dfs2(int x,int t){
top[x]=t;
dfn[x]=++tim;
if(son[x]) dfs2(son[x],t);
for(int i=head[x];i;i=E[i].next){
int y=E[i].to;
if(y!=fa[x]&&y!=son[x]){
dfs2(y,y);
}
}
}
int lca(int x,int y){
while(top[x]!=top[y]){
if(deep[top[x]]>deep[top[y]]) x=fa[top[x]];
else y=fa[top[y]];
}
if(deep[x]<deep[y]) return x;
else return y;
} void update_point(int x,long long uval,segment_tree &T){
T.update(dfn[x],uval,1);
}
long long query_route(int x,int y,segment_tree &T){
int tx=top[x];
int ty=top[y];
long long ans=0;
while(tx!=ty){
if(deep[tx]<deep[ty]){
swap(tx,ty);
swap(x,y);
}
ans+=T.query(dfn[tx],dfn[x],1);
x=fa[tx];
tx=top[x];
}
if(deep[x]>deep[y]) swap(x,y);
ans+=T.query(dfn[x],dfn[y],1);
return ans;
} struct route{
int x;
int y;
int lc;
long long w;
}q[maxm+5];
vector<int>rid[maxn+5];//两端点lca为x的链的编号 long long sum[maxn+5];//sum(dp[son(x)])
long long dp[maxn+5];
void dfs3(int x){//dp过程
sum[x]=0;
for(int i=head[x];i;i=E[i].next){
int y=E[i].to;
if(y!=fa[x]){
dfs3(y);
sum[x]+=dp[y];
}
}
dp[x]=sum[x];
update_point(x,sum[x],T1);//因为下一步就要用到,这里要更新sum[x]
for(int i=0;i<rid[x].size();i++){
int k=rid[x][i];
dp[x]=max(dp[x],q[k].w+query_route(q[k].x,q[k].y,T1)-query_route(q[k].x,q[k].y,T2));
}
update_point(x,dp[x],T2);
} void ini(){
memset(deep,0,sizeof(deep));
memset(dfn,0,sizeof(dfn));
memset(dp,0,sizeof(dp));
memset(fa,0,sizeof(fa));
memset(head,0,sizeof(head));
for(int i=1;i<=n;i++) rid[i].clear();
memset(son,0,sizeof(son));
memset(sum,0,sizeof(sum));
esz=1;
memset(sz,0,sizeof(sz));
tim=0;
memset(top,0,sizeof(top));
T1.build(1,n,1);
T2.build(1,n,1);
}
int main(){
int u,v,w;
scanf("%d",&t);
while(t--){
scanf("%d %d",&n,&m);
ini();
for(int i=1;i<n;i++){
scanf("%d %d",&u,&v);
add_edge(u,v);
add_edge(v,u);
}
dfs1(1,0);
dfs2(1,1);
for(int i=1;i<=m;i++){
scanf("%d %d %lld",&q[i].x,&q[i].y,&q[i].w);
q[i].lc=lca(q[i].x,q[i].y);
rid[q[i].lc].push_back(i);
}
dfs3(1);
printf("%lld\n",dp[1]);
} }

[HDU 5293]Tree chain problem(树形dp+树链剖分)的更多相关文章

  1. HDU 5293 Tree chain problem 树形dp&plus;dfs序&plus;树状数组&plus;LCA

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5293 题意: 给你一些链,每条链都有自己的价值,求不相交不重合的链能够组成的最大价值. 题解: 树形 ...

  2. HDU 5293 Tree chain problem 树形DP

    题意: 给出一棵\(n\)个节点的树和\(m\)条链,每条链有一个权值. 从中选出若干条链,两两不相交,并且使得权值之和最大. 分析: 题解 #include <cstdio> #incl ...

  3. &lpar;中等&rpar; HDU 5293 Tree chain problem,树链剖分&plus;树形DP。

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

  4. hdu5293 Tree chain problem 树形dp&plus;线段树

    题目:pid=5293">http://acm.hdu.edu.cn/showproblem.php?pid=5293 在一棵树中,给出若干条链和链的权值.求选取不相交的链使得权值和最 ...

  5. HDU 5293 Tree chain problem

    树状数组 + dp 设$f_i$表示以$i$为根的子树中的能选取的最大和,$sum_x$表示$\sum_{f_y}$  ($y$是$x$的一个儿子),这样子我们把所有给出的链按照两点的$lca$分组, ...

  6. codeforces 671D Roads in Yusland &amp&semi; hdu 5293 Tree chain problem

    dp dp优化 dfs序 线段树 算是一个套路.可以处理在树上取链的问题.

  7. 6&period;3 省选模拟赛 Decompose 动态dp 树链剖分 set

    LINK:Decompose 看起来很难 实际上也很难 考验选手的dp 树链剖分 矩阵乘法的能力. 容易列出dp方程 暴力dp 期望得分28. 对于链的情况 容易发现dp方程可以转矩阵乘法 然后利用线 ...

  8. HDU 5293 Train chain Problem - 树链剖分&lpar;树状数组&rpar; &plus; 线段树&plus; 树型dp

    传送门 题目大意: 一颗n个点的树,给出m条链,第i条链的权值是\(w_i\),可以选择若干条不相交的链,求最大权值和. 题目分析: 树型dp: dp[u][0]表示不经过u节点,其子树的最优值,dp ...

  9. HDU 3966:Aragorn&&num;39&semi;s Story(树链剖分)

    http://acm.hdu.edu.cn/showproblem.php?pid=3966 题意:有n个点n-1条边,每个点有一个权值,有两种操作:询问一个点上权值是多少和修改u到v这条链上的权值. ...

随机推荐

  1. 无法启动ArcSDE服务

    ArcSDE服务启动错误:Error (-327), No ArcSDE server license found解决方法:>sdesetup -o update_key -d oracle10 ...

  2. VB&period;net 利用SerialPort进行读取串口操作

    Imports SystemImports System.IO.Ports Public Class Form1 Private Sub Form1_Load(ByVal sender As Syst ...

  3. poj1704

    题目大意:n个石子,每次可以取一个石子向左移动,左边有边界限制,每个位置最多同时放一个石子,求先手必胜还是必败. 首先,我们将石子两两配对,每一对,若是先手可以将左边的向左移动一格,则可以用后手将右边 ...

  4. 登陆权限验证Session和Cookie用法及BasePage类使用

    最近在做ASP.NET的项目时,接触到了登陆权限模块,所有总结了一下登陆时用到的知识和方法技巧. 如图说明:实现的效果如图,由于验证码验证比较简单这里就不介绍了 首先用代码生成器生成项目,以三层为例进 ...

  5. &lbrack;题解&rsqb;bzoj 1861 Book 书架 - Splay

    1861: [Zjoi2006]Book 书架 Time Limit: 4 Sec  Memory Limit: 64 MBSubmit: 1396  Solved: 803[Submit][Stat ...

  6. &lbrack;原创&rsqb;Hadoop-2&period;5&period;2-HA原文译

    使用the Quorum Journal Manager实现HDFS高可用 2017/1/22 11:57:22 原文 目的(Purpose) * 这个指南提供了对HDFS-HA特性,使用QJM特性如 ...

  7. MySQL读写分离技术

    1.简介 当今MySQL使用相当广泛,随着用户的增多以及数据量的增大,高并发随之而来.然而我们有很多办法可以缓解数据库的压力.分布式数据库.负载均衡.读写分离.增加缓存服务器等等.这里我们将采用读写分 ...

  8. nginx &plus; tomcat实现负载均衡

    作者Mr.Chen,转载请注明博客出处:http://www.cnblogs.com/cjh-notes/ 负载均衡 负载均衡就是流量分发,优选软件解决方案,成本低效果好. 实现步骤 第一步:下载安装 ...

  9. docker-machine on azure

    1.准备Azure的虚拟机,安装docker-machine 由于azure虚拟机的管理员账号不是root,所以这里我们使用自己创建的管理员yy 1.base=https://github.com/d ...

  10. Matplotlib python 基本用法

    1.简单的绘制函数 import matplotlib.pyplot as plt import numpy as np x = np.linspace(-1, 1, 50) y1 = x + 1 p ...