hdu5293 lca+dp+树状数组+时间戳
题意是给了 n 个点的树,会有m条链条 链接两个点,计算出他们没有公共点的最大价值, 公共点时这样计算的只要在他们 lca 这条链上有公共点的就说明他们相交
dp[i]为这个点包含的子树所能得到的最大价值
sum[i]表示这个点没有选择经过i这个点链条的总价值
两种选择
这个点没有被选择
dp[i]=sum[i]=sigma(dp[k])k为i的子树
选择了某个链
假设这条链 为(tyuijk)
那么dp[i]=(sum[i]-dp[u]-dp[j])+(sum[j]-dp[k])+dp[k] +(sum[u]-dp[y])+(sum[y]-dp[t])+sum[t];
整理后发现 dp[i]=sum[i] +(sum[j]-dp[j])+(sum[k]-dp[k])+(sum[u]-dp[u])+(sum[y]-dp[y])+(sum[t]-dp[t]);
使用lca计算出每条链的最近公共祖先,在这个最近公共祖先上判断是否使用这条链,还有我们可以使用时间戳加树状数组来求得sum和dp
#include <iostream>
#include <algorithm>
#include <string.h>
#include <cstdio>
#include <vector>
using namespace std;
const int maxn=+;
int to[maxn*],nx[maxn*],H[maxn*],numofedg,timoflook;
int fa[maxn][],first[maxn],last[maxn],depth[maxn];
void addedg(int u, int v)
{
numofedg++; to[numofedg]=v; nx[numofedg]=H[u]; H[u]=numofedg;
numofedg++; to[numofedg]=u; nx[numofedg]=H[v]; H[v]=numofedg;
}
void dfs(int cur, int per, int dep)
{
first[cur]=++timoflook;
depth[cur]=dep;
fa[cur][]=per;
for(int i=; i<; i++)
{
fa[cur][i]=fa[ fa[cur][i-] ][ i- ];
}
for(int i=H[cur]; i; i=nx[i])
{
if(to[i]==per)continue;
dfs(to[i],cur,dep+);
}
last[cur]=++timoflook;
}
int getlca(int u,int v)
{
if(depth[u]<depth[v])swap(u,v);
for(int i=; i>=; i--)
{
if(depth[fa[u][i]]>=depth[v])
u=fa[u][i];
if(u==v)return u;
}
for(int i=; i>=; i--)
{
if(fa[u][i]!=fa[v][i])
{
u=fa[u][i];
v=fa[v][i];
}
}
return fa[u][];
}
struct Edg
{
int u,v,lca,val;
}P[maxn];
vector<int>E[maxn];
int dp[maxn],sum[maxn],CS[maxn*],CD[maxn*];
int lowbit(int x)
{
return x&-x;
}
void add(int x, int d, int *C)
{
while(x<=timoflook)
{
C[x]+=d;
x+=lowbit(x);
}
}
int getsum(int x, int *C)
{
int ret=;
while(x>)
{
ret+=C[x];
x-=lowbit(x);
}
return ret;
}
void solve(int cur, int per)
{
dp[cur]=sum[cur]=;
for(int i=H[cur]; i; i=nx[i])
{
if(to[i]==per)continue;
solve(to[i],cur);
sum[cur]+=dp[to[i]];
}
dp[cur]=sum[cur];
for(int i=; i<E[cur].size(); i++)
{
int id=E[cur][i];
int u=P[id].u;
int v=P[id].v;
int t1=getsum(first[u],CS);
int t2=getsum(first[v],CS);
int t3=getsum(first[u],CD);
int t4=getsum(first[v],CD);
int tmp=t1+t2-t3-t4;
dp[cur]=max(dp[cur],tmp+P[id].val+sum[cur]);
}
add(first[cur],sum[cur],CS);
add(last[cur],-sum[cur],CS);
add(first[cur],dp[cur],CD);
add(last[cur],-dp[cur],CD); }
int main()
{
int cas;
scanf("%d",&cas);
for(int cc=; cc<=cas; cc++)
{
int n,m;
timoflook=numofedg=;
scanf("%d%d",&n,&m);
for(int i=; i<=n; i++)
{
CS[i*]=CS[i*+]=CD[i*]=CD[i*+]=;
H[i]=;E[i].clear(); } for(int i=; i<n; i++)
{
int u,v;
scanf("%d%d",&u,&v);
addedg(u,v);
}
fa[][]=;
dfs(,,);
for(int i=; i<m; i++)
{
scanf("%d%d%d",&P[i].u,&P[i].v,&P[i].val);
P[i].lca=getlca(P[i].u,P[i].v);
E[P[i].lca].push_back(i);
}
solve(,-);
printf("%d\n",dp[]);
}
return ;
}
hdu5293 lca+dp+树状数组+时间戳的更多相关文章
- 树形DP+树状数组 HDU 5877 Weak Pair
//树形DP+树状数组 HDU 5877 Weak Pair // 思路:用树状数组每次加k/a[i],每个节点ans+=Sum(a[i]) 表示每次加大于等于a[i]的值 // 这道题要离散化 #i ...
- bzoj 1264 [AHOI2006]基因匹配Match(DP+树状数组)
1264: [AHOI2006]基因匹配Match Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 793 Solved: 503[Submit][S ...
- Luogu 2680 NOIP 2015 运输计划(树链剖分,LCA,树状数组,树的重心,二分,差分)
Luogu 2680 NOIP 2015 运输计划(树链剖分,LCA,树状数组,树的重心,二分,差分) Description L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之 ...
- 【bzoj2274】[Usaco2011 Feb]Generic Cow Protests dp+树状数组
题目描述 Farmer John's N (1 <= N <= 100,000) cows are lined up in a row andnumbered 1..N. The cows ...
- POJ 2763 (LCA +RMQ+树状数组 || 树链部分) 查询两点距离+修改边权
题意: 知道了一颗有 n 个节点的树和树上每条边的权值,对应两种操作: 0 x 输出 当前节点到 x节点的最短距离,并移动到 x 节点位置 1 x val 把第 x 条边的权值改为 ...
- 奶牛抗议 DP 树状数组
奶牛抗议 DP 树状数组 USACO的题太猛了 容易想到\(DP\),设\(f[i]\)表示为在第\(i\)位时方案数,转移方程: \[ f[i]=\sum f[j]\;(j< i,sum[i] ...
- HDU5293(SummerTrainingDay13-B Tree DP + 树状数组 + dfs序)
Tree chain problem Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Other ...
- UESTC 912 树上的距离 --LCA+RMQ+树状数组
1.易知,树上两点的距离dis[u][v] = D[u]+D[v]-2*D[lca(u,v)] (D为节点到根节点的距离) 2.某条边<u,v>权值一旦改变,将会影响所有以v为根的子树上的 ...
- codeforces 597C C. Subsequences(dp+树状数组)
题目链接: C. Subsequences time limit per test 1 second memory limit per test 256 megabytes input standar ...
随机推荐
- 使用U盘为虚拟机安装系统
前提:使用虚拟机安装WIN8系统时,由于WIN8镜像文件大于4G无法使用虚拟安装,所以使用U盘安装. 1.装有U盘启动的WINPe系统 (1)下载 老毛桃U盘启动盘制作工具 (2)U盘清空 2.虚拟机 ...
- jquery 数组的操作
1.数组的创建 var arrayObj = new Array(); //创建一个数组 var arrayObj = new Array([size]); //创建一个数组并指定长度,注意不是上限, ...
- Orchard Core 版本冲突 The type 'FormTagHelper' exists in both 'Microsoft.AspNetCore.Mvc.TagHelpers, Version=2.1.1.0, Culture=neutral, PublicKeyToken=adb9793829ddae60' and...
最近老大让我看Orchard Core,这是一个CMS系统.可以先参考大佬的文章:https://www.cnblogs.com/shanyou/archive/2018/09/25/9700422. ...
- SQL查询优化:详解SQL Server非聚集索引(转载)
本文是转载,原文地址 http://tech.it168.com/a2011/1228/1295/000001295176.shtml 在SQL SERVER中,非聚集索引其实可以看作是一个含有聚集索 ...
- C# cmd bcp 导出数据
背景需求:应用系统间数据自动同步处理,要求高效无人工干预 技术实现:C#启动cmd,通过BCP命令传入必要参数,实现数据导出 /// <summary> /// cmd下,启动应用程序命令 ...
- 什么是Docker?(一)
Docker 最初是 dotCloud 公司创始人 Solomon Hykes 在法国期间发起的一个公司内部项目,它是基于 dotCloud 公司多年云服务技术的一次革新,并于 2013 年 3 月以 ...
- navicat如何导出mysql数据表结构
我们在创建数据库时会对字段进行设置,比如类型.长度等,如果字段多的话一个个设置非常麻烦,可以从其他地方已有的表导入数据表结构,怎么操作呢?我们拿navicat导出mysql数据表结构为例: 1.点击“ ...
- 电脑出现 flash update failed 解决方法
笔记本电脑过了一个周末打开时出现以上问题,每次都进入这个界面 解决方法: 拆机,插拨一下内存条,硬盘,就启动了
- what's the 回撤
什么是“回撤”? “回撤”是个谓语,前面隐含了一个主语.一般来说,没有人说“亏损回撤”的,我们说的“回撤”,通常指“股价回撤”.“市值回撤”.“净值回撤”和“盈利回撤”. “股价回撤”是针对个股的,即 ...
- linux服务器性能查看
1.1 cpu性能查看 1.查看物理cpu个数: cat /proc/cpuinfo |grep "physical id"|sort|uniq|wc -l 2.查看每个物理cpu ...