题意是给了 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+树状数组+时间戳的更多相关文章

  1. 树形DP+树状数组 HDU 5877 Weak Pair

    //树形DP+树状数组 HDU 5877 Weak Pair // 思路:用树状数组每次加k/a[i],每个节点ans+=Sum(a[i]) 表示每次加大于等于a[i]的值 // 这道题要离散化 #i ...

  2. bzoj 1264 [AHOI2006]基因匹配Match(DP+树状数组)

    1264: [AHOI2006]基因匹配Match Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 793  Solved: 503[Submit][S ...

  3. Luogu 2680 NOIP 2015 运输计划(树链剖分,LCA,树状数组,树的重心,二分,差分)

    Luogu 2680 NOIP 2015 运输计划(树链剖分,LCA,树状数组,树的重心,二分,差分) Description L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之 ...

  4. 【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 ...

  5. POJ 2763 (LCA +RMQ+树状数组 || 树链部分) 查询两点距离+修改边权

    题意: 知道了一颗有  n 个节点的树和树上每条边的权值,对应两种操作: 0 x        输出 当前节点到 x节点的最短距离,并移动到 x 节点位置 1 x val   把第 x 条边的权值改为 ...

  6. 奶牛抗议 DP 树状数组

    奶牛抗议 DP 树状数组 USACO的题太猛了 容易想到\(DP\),设\(f[i]\)表示为在第\(i\)位时方案数,转移方程: \[ f[i]=\sum f[j]\;(j< i,sum[i] ...

  7. HDU5293(SummerTrainingDay13-B Tree DP + 树状数组 + dfs序)

    Tree chain problem Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Other ...

  8. UESTC 912 树上的距离 --LCA+RMQ+树状数组

    1.易知,树上两点的距离dis[u][v] = D[u]+D[v]-2*D[lca(u,v)] (D为节点到根节点的距离) 2.某条边<u,v>权值一旦改变,将会影响所有以v为根的子树上的 ...

  9. codeforces 597C C. Subsequences(dp+树状数组)

    题目链接: C. Subsequences time limit per test 1 second memory limit per test 256 megabytes input standar ...

随机推荐

  1. kubernetes微服务部署

    1.哪些服务适合单独成为一个pod?哪些服务适合在一个pod中? message消息服务被很多服务调用   单独一个pod dubbo服务和web服务交互很高放在同一个pod里 API网关调用很多服务 ...

  2. 使用Git,如何忽略不需要上传的文件(配置文件)

    步骤1:在目录下,选择GIt Bash Here 2.输入命令 : git update-index --assume-unchanged 文件名 3.再输入指令 git  status 查看修改文件 ...

  3. java JDBC (八) 连接池 DBCP

    package cn.sasa.demo1; import javax.sql.DataSource; import org.apache.commons.dbcp2.BasicDataSource; ...

  4. 【PyQt5-Qt Designer】在GUI中使用pyqtgraph绘图库

    pyqtgraph绘图库 1.1 简介: pyqtgraph是Python平台上一种功能强大的2D/3D绘图库,相对于matplotlib库,由于内部实现方式上,使用了高速计算的numpy信号处理库以 ...

  5. 使用jquery.uploadify上传文件

    今天在网上找了一天,想要找到一个比较全的使用案例,结果发现基本上全是一个版本的... 我的问题主要是上传完成后,还需要将路径获取到,然后保存到数据库. 查了一下资料发现有这么一个参数onComplet ...

  6. SpringBoot-整合多数据源

    整合多数据源 这里有两种,分包数据源和注解数据源,这里讲分包数据源 配置文件中新增两个数据源 spring.datasource.test1.driverClassName = com.mysql.j ...

  7. webstorm 2017激活

    选择“license server” 输入:http://idea.imsxm.com/

  8. Python3学习之路~6.7 经典类和新式类的继承顺序

    在Python中,经典类(class Person:)和新式类(class Person(object):)的主要区别就是体现在多继承的顺序上. Python 2.x中默认都是经典类,只有显式继承了o ...

  9. Mysql聚集索引的使用

    聚集索引 聚簇索引并不是一种单独的索引类型,而是一种数据存储方式(不是数据结构,而是存储结构),具体细节依赖于其实现方式,聚簇索引实际上是在同一个结构中保存了btree索引和数据行. innodb将通 ...

  10. 清空select标签中option选项的3种不同方式

    方法一 代码如下:document.getElementById("selectid").options.length = 0; 方法二 代码如下:document.formNam ...