Description

小B最近正在玩一个寻宝游戏,这个游戏的地图中有N个村庄和N-1条道路,并且任何两个村庄之间有且仅有一条路径可达。游戏开始时,玩家可以任意选择一个村庄,瞬间转移到这个村庄,然后可以任意在地图的道路上行走,若走到某个村庄中有宝物,则视为找到该村庄内的宝物,直到找到所有宝物并返回到最初转移到的村庄为止。小B希望评测一下这个游戏的难度,因此他需要知道玩家找到所有宝物需要行走的最短路程。但是这个游戏中宝物经常变化,有时某个村庄中会突然出现宝物,有时某个村庄内的宝物会突然消失,因此小B需要不断地更新数据,但是小B太懒了,不愿意自己计算,因此他向你求助。为了简化问题,我们认为最开始时所有村庄内均没有宝物

Input

第一行,两个整数N、M,其中M为宝物的变动次数。

接下来的N-1行,每行三个整数x、y、z,表示村庄x、y之间有一条长度为z的道路。
接下来的M行,每行一个整数t,表示一个宝物变动的操作。若该操作前村庄t内没有宝物,则操作后村庄内有宝物;若该操作前村庄t内有宝物,则操作后村庄内没有宝物。

Output

M行,每行一个整数,其中第i行的整数表示第i次操作之后玩家找到所有宝物需要行走的最短路程。若只有一个村庄内有宝物,或者所有村庄内都没有宝物,则输出0。

Sample Input

4 5
1 2 30
2 3 50
2 4 60
2
3
4
2
1

Sample Output

0
100
220
220
280

HINT

1<=N<=100000

1<=M<=100000
对于全部的数据,1<=z<=10^9

  这道题挺经典的,做法是按DFS序重标号,然后:

    一.加入一个点b,找到有宝物的集合中重标号在其左边最近的一个和右边最近的一个,左边若没有则找到右边最远的一个,右边没有则找到左边最远的一个,记为a和b,答案加上lca(a,b)+lca(b,c)-lca(a,b),不难发现这样是正确的.

    二.删除一个点b,同样方法找到a,c,答案减去lca(a,b)+lca(b,c)-lca(a,b),类似。

    还有集合的维护其实可以用STL::set,我用的BIT+二分,代码量大了一点……

 #include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
const int maxn=;
long long ans;
long long dis[maxn][];
int cnt,fir[maxn],to[maxn<<],nxt[maxn<<],val[maxn<<];
int n,m,mm,fa[maxn][],dep[maxn],bit[maxn];
void addedge(int a,int b,long long d){
nxt[++cnt]=fir[a];to[cnt]=b;val[cnt]=d;fir[a]=cnt;
}
void DFS(int node){
for(int i=fir[node];i;i=nxt[i]){
if(to[i]==fa[node][])continue;
fa[to[i]][]=node;
dis[to[i]][]=val[i];
dep[to[i]]=dep[node]+;
DFS(to[i]);
}
} void Init(){
for(int k=;k<=mm;k++)
for(int i=;i<=n;i++)
fa[i][k]=fa[fa[i][k-]][k-],
dis[i][k]=dis[i][k-]+dis[fa[i][k-]][k-];
} int ntp[maxn],ptn[maxn],cont; void DFS2(int node){
ntp[node]=++cont;ptn[cont]=node;
for(int i=fir[node];i;i=nxt[i])
if(to[i]!=fa[node][])
DFS2(to[i]);
} int Query(int x){
int ret=;
while(x){
ret+=bit[x];
x-=x&(-x);
}
return ret;
} void add(int x,int d){
while(x<=n){
bit[x]+=d;
x+=x&(-x);
}
} long long Lca(int x,int y){
long long ret=;
if(dep[x]<dep[y])swap(x,y);
for(int i=mm,d=dep[x]-dep[y];i>=;i--)
if(d&(<<i))
ret+=dis[x][i],x=fa[x][i]; for(int i=mm;x!=y;i?i--:i){
if(!i||fa[x][i]!=fa[y][i])
ret+=dis[x][i]+dis[y][i],
x=fa[x][i],y=fa[y][i];
}
return ret;
} int Ql(int l,int r){
if(Query(r)-Query(l-)==)return r+;
while(l<r){
int mid=(l+r)>>;
if(Query(mid)-Query(l-))r=mid;
else l=mid+;
}
return l;
} int Qr(int l,int r){
if(Query(r)-Query(l-)==)return l-;
while(l<r){
int mid=(l+r+)>>;
if(Query(r)-Query(mid-))l=mid;
else r=mid-;
}
return r;
} int main(){
scanf("%d%d",&n,&m);
while(<<(mm+)<=n)mm++;
for(int i=;i<n;i++){
int a,b,v;
scanf("%d%d%d",&a,&b,&v);
addedge(a,b,v);
addedge(b,a,v);
}
DFS();Init();DFS2(); int p,j,k;
while(m--){
scanf("%d",&p);
if(Query(ntp[p])-Query(ntp[p]-)){
add(ntp[p],-);
j=Qr(,ntp[p]-);
k=Ql(ntp[p]+,n);
if(j==&&k==n+);
else{
if(j==) j=Qr(ntp[p]+,n);
if(k==n+) k=Ql(,ntp[p]-);
ans-=Lca(ptn[j],p)+Lca(ptn[k],p)-Lca(ptn[j],ptn[k]);
}
}
else{
add(ntp[p],);
j=Qr(,ntp[p]-);
k=Ql(ntp[p]+,n);
if(j==&&k==n+);
else{
if(j==) j=Qr(ntp[p]+,n);
if(k==n+) k=Ql(,ntp[p]-);
ans+=Lca(ptn[j],p)+Lca(ptn[k],p)-Lca(ptn[j],ptn[k]);
}
}
printf("%lld\n",ans);
}
return ;
}

树形结构的维护:BZOJ 3991: [SDOI2015]寻宝游戏的更多相关文章

  1. bzoj 3991: [SDOI2015]寻宝游戏 虚树 set

    目录 题目链接 题解 代码 题目链接 bzoj 3991: [SDOI2015]寻宝游戏 题解 发现每次答案就是把虚树上的路径*2 接在同一关键点上的点的dfs序是相邻的 那么用set动态维护dfs序 ...

  2. bzoj 3991: [SDOI2015]寻宝游戏

    Description 小B最近正在玩一个寻宝游戏,这个游戏的地图中有N个村庄和N-1条道路,并且任何两个村庄之间有且仅有一条路径可达.游戏开始时,玩家可以任意选择一个村庄,瞬间转移到这个村庄,然后可 ...

  3. [BZOJ 3991][SDOI2015]寻宝游戏(dfs序)

    题面 小B最近正在玩一个寻宝游戏,这个游戏的地图中有N个村庄和N-1条道路,并且任何两个村庄之间有且仅有一条路径可达.游戏开始时,玩家可以任意选择一个村庄,瞬间转移到这个村庄,然后可以任意在地图的道路 ...

  4. BZOJ 3991: [SDOI2015]寻宝游戏 树链的并+set

    Description 小B最近正在玩一个寻宝游戏,这个游戏的地图中有N个村庄和N-1条道路,并且任何两个村庄之间有且仅有一条路径可达.游戏开始时,玩家可以任意选择一个村庄,瞬间转移到这个村庄,然后可 ...

  5. BZOJ 3991: [SDOI2015]寻宝游戏 [虚树 树链的并 set]

    传送门 题意: $n$个点的树,$m$次变动使得某个点有宝物或没宝物,询问每次变动后集齐所有宝物并返回原点的最小距离 转化成有根树,求树链的并... 两两树链求并就可以,但我们按照$dfs$序来两两求 ...

  6. BZOJ.3991.[SDOI2015]寻宝游戏(思路 set)

    题目链接 从哪个点出发最短路径都是一样的(最后都要回来). 脑补一下,最短路应该是按照DFS的顺序,依次访问.回溯遍历所有点,然后再回到起点. 即按DFS序排序后,Ans=dis(p1,p2)+dis ...

  7. 3991: [SDOI2015]寻宝游戏

    3991: [SDOI2015]寻宝游戏 https://www.lydsy.com/JudgeOnline/problem.php?id=3991 分析: 虚树+set. 要求树上许多点之间的路径的 ...

  8. 【BZOJ】3991: [SDOI2015]寻宝游戏

    题意 给一个\(n\)个点带边权的树.有\(m\)次操作,每一次操作一个点\(x\),如果\(x\)已经出现,则\(x\)消失.否则\(x\)出现.每一操作后,询问从某个点开始走,直到经过所有出现的点 ...

  9. 【BZOJ】3991: [SDOI2015]寻宝游戏 虚树+DFS序+set

    [题意]给定n个点的带边权树,对于树上存在的若干特殊点,要求任选一个点开始将所有特殊点走遍后返回.现在初始没有特殊点,m次操作每次增加或减少一个特殊点,求每次操作后的总代价.n,m<=10^5. ...

随机推荐

  1. JS快速排序和去重

    JS的快速排序和JS去重在面试的时候问的挺多的.下面是我对快速排序的理解,和快速排序,去重的代码. 1.什么是快速排序? 第一步: 快速排序就是去个中间值,把比中间值小的放在左边设为arrLeft,比 ...

  2. C# 内存管理优化畅想(三)---- 其他方法&结语

    前两篇文章提出的优化方法,都是不需要修改源代码的,而是在CLR或JIT层面进行自动优化的.但本文中提出的优化方法则需要引入新的语法,开发者只有在源代码中使用了这些新语法,才会获得优化. 1. 允许对象 ...

  3. (tomcat访问不了的两种解决方法)Bad Request(Invalid Hostname)

    显示这个页面的时候一般有几中解决方法: 第一种就是如下图所示的方法: 具体步骤是: 1.也就是左下角win的“运行”中输入cmd进入doc窗口中 2.输入代码:netstat -ano 3.找到占用8 ...

  4. 文字排版--删除线(text-decoration:line-through)

    如果想在网页上设置删除线怎么办,这个样式在电商网站上常会见到: 上图中的原价上的删除线使用下面代码就可以实现: .oldPrice{text-decoration:line-through;}

  5. [转]Delphi中ShellExecute的妙用

    Delphi中ShellExecute的妙用       ShellExecute的功能是运行一个外部程序(或者是打开一个已注册的文件.打开一个目录.打印一个文件等等),并对外部程序有一定的控制.   ...

  6. Linux查看进程内存占用及内存使用情况

    LINUX进程内存占用查看方法(1)top可以直接使用top命令后,查看%MEM的内容.可以选择按进程查看或者按用户查看,如想查看oracle用户的进程内存使用情况的话可以使用如下的命令:$ top ...

  7. SGU 111.Very simple problem

    题目大意:              求平方不大于n(n<=10^1000)的最大的数. 分析:              二分+高精度乘法 或者 高精度开方...               ...

  8. newInstance()和new的区别

    在初始化一个类,生成一个实例的时候:newInstance() 和 new 有什么区别?用newInstance与用new是有区别的,区别在于创建对象的方式不一样,前者是使用类加载机制,那么为什么会有 ...

  9. 原生Ajax + Promise

    有原生写的ajax + promise嫁接下 ;(function(root){ var LD = function(obj){ if( obj instanceof LD ) return obj; ...

  10. Ajax实现的长轮询不阻塞同一时间内页面的其他Ajax请求(同域请求)

    最近要做一个来电的弹屏功能,利用OM 系统的接口,OM系统发送请求到接口程序,分析数据添加到mysql数据库中,然后把最新的数据id 跟今日来电的总的数量存储到memcache 中.弹屏程序根据读取的 ...