P3320 [SDOI2015]寻宝游戏 解题报告
P3320 [SDOI2015]寻宝游戏
题目描述
小B最近正在玩一个寻宝游戏,这个游戏的地图中有\(N\)个村庄和\(N-1\)条道路,并且任何两个村庄之间有且仅有一条路径可达。游戏开始时,玩家可以任意选择一个村庄,瞬间转移到这个村庄,然后可以任意在地图的道路上行走,若走到某个村庄中有宝物,则视为找到该村庄内的宝物,直到找到所有宝物并返回到最初转移到的村庄为止。
小B希望评测一下这个游戏的难度,因此他需要知道玩家找到所有宝物需要行走的最短路程。但是这个游戏中宝物经常变化,有时某个村庄中会突然出现宝物,有时某个村庄内的宝物会突然消失,因此小\(B\)需要不断地更新数据,但是小\(B\)太懒了,不愿意自己计算,因此他向你求助。为了简化问题,我们认为最开始时所有村庄内均没有宝物
输入输出格式
输入格式:
第一行,两个整数\(N\)、\(M\),其中\(M\)为宝物的变动次数。接下来的\(N-1\)行,每行三个整数\(x\)、\(y\)、\(z\),表示村庄\(x\)、\(y\)之间有一条长度为z的道路。接下来的\(M\)行,每行一个整数\(t\),表示一个宝物变动的操作。若该操作前村庄\(t\)内没有宝物,则操作后村庄内有宝物;若该操作前村庄t内有宝物,则操作后村庄内没有宝物。
输出格式:
\(M\)行,每行一个整数,其中第\(i\)行的整数表示第\(i\)次操作之后玩家找到所有宝物需要行走的最短路程。若只有一个村庄内有宝物,或者所有村庄内都没有宝物,则输出\(0\)。
说明
\(1<=N<=100000\)
\(1<=M<=100000\)
对于全部的数据,\(1<=z<=10^9\)
首先发现似乎走两次,然而没啥用,然后yy一波,也没啥用。
考虑走出了一个环,想想虚树的构建,如果按\(dfs\)序走这个环,那么就可以了。
为什么呢,我只会意会...
然后用\(set\)维护一下\(dfs\)序就行了,边界写起来怪麻烦的。
Code:
#include <cstdio>
#include <set>
#define ll long long
const int N=1e5+10;
int head[N],to[N<<1],Next[N<<1],edge[N<<1],cnt;
void add(int u,int v,int w)
{
to[++cnt]=v,Next[cnt]=head[u],edge[cnt]=w,head[u]=cnt;
}
int dfn[N],f[N][18],dep[N],ha[N],dfsclock;
ll dis[N];
void dfs(int now)
{
dep[now]=dep[f[now][0]]+1;
ha[dfn[now]=++dfsclock]=now;
for(int i=1;f[now][i-1];i++) f[now][i]=f[f[now][i-1]][i-1];
for(int v,i=head[now];i;i=Next[i])
if((v=to[i])!=f[now][0])
dis[v]=dis[now]+edge[i],f[v][0]=now,dfs(v);
}
int LCA(int x,int y)
{
if(dep[x]<dep[y]) return LCA(y,x);
for(int i=17;~i;i--)
if(dep[f[x][i]]>=dep[y])
x=f[x][i];
if(x==y) return x;
for(int i=17;~i;i--)
if(f[x][i]!=f[y][i])
x=f[x][i],y=f[y][i];
return f[x][0];
}
ll Dis(int x,int y){return dis[x]+dis[y]-(dis[LCA(x,y)]<<1);}
std::set <int> s;
std::set <int>::iterator it;
ll sum;int n,m,tag[N];
int main()
{
scanf("%d%d",&n,&m);
for(int u,v,w,i=1;i<n;i++) scanf("%d%d%d",&u,&v,&w),add(u,v,w),add(v,u,w);
dfs(1);
for(int t,i=1;i<=m;i++)
{
scanf("%d",&t);
if(tag[t])//删
{
if(s.size()!=1)
{
it=s.lower_bound(dfn[t]);
if(it==s.begin())
{
++it;
sum-=Dis(t,ha[*it]);
sum-=Dis(t,ha[*--s.end()]);
sum+=Dis(ha[*it],ha[*--s.end()]);
}
else if(it==--s.end())
{
sum-=Dis(t,ha[*--it]);
sum-=Dis(t,ha[*s.begin()]);
sum+=Dis(ha[*it],ha[*s.begin()]);
}
else
{
int p=*--it;
sum-=Dis(t,ha[p]);
++it,++it;
sum-=Dis(t,ha[*it]);
sum+=Dis(ha[p],ha[*it]);
}
}
s.erase(dfn[t]);
}
else
{
if(!s.empty())
{
it=s.lower_bound(dfn[t]);
if(it==s.end())
{
sum+=Dis(t,ha[*--it]);
sum+=Dis(t,ha[*s.begin()]);
sum-=Dis(ha[*it],ha[*s.begin()]);
}
else if(it==s.begin())
{
sum+=Dis(t,ha[*it]);
sum+=Dis(t,ha[*--s.end()]);
sum-=Dis(ha[*it],ha[*--s.end()]);
}
else
{
int p=*it;
sum+=Dis(t,ha[p]);
--it;
sum+=Dis(t,ha[*it]);
sum-=Dis(ha[p],ha[*it]);
}
}
s.insert(dfn[t]);
}
tag[t]^=1;
printf("%lld\n",sum);
}
return 0;
}
2018.12.14
P3320 [SDOI2015]寻宝游戏 解题报告的更多相关文章
- P3320 [SDOI2015]寻宝游戏
题目 P3320 [SDOI2015]寻宝游戏 做法 很巧妙的一种思路,懂了之后觉得大水题 首先要知道:在一棵树上标记一些点,然后从任意一点出发,遍历所有的的最小路径为\(dfs\)序从小到大遍历 那 ...
- Luogu P3320 [SDOI2015]寻宝游戏 / 异象石 【LCA/set】
期末考试结束祭! 在期末考试前最后一发的测试中,异象石作为第二道题目出现QAQ.虽然知道是LCA图论,但还是敲不出来QAQ. 花了两天竞赛课的时间搞懂(逃 异象石(stone.pas/c/cpp)题目 ...
- [洛谷P3320] SDOI2015 寻宝游戏
问题描述 小B最近正在玩一个寻宝游戏,这个游戏的地图中有N个村庄和N-1条道路,并且任何两个村庄之间有且仅有一条路径可达.游戏开始时,玩家可以任意选择一个村庄,瞬间转移到这个村庄,然后可以任意在地图的 ...
- [bzoj3991] [洛谷P3320] [SDOI2015] 寻宝游戏
Description 小B最近正在玩一个寻宝游戏,这个游戏的地图中有 \(N\) 个村庄和 \(N-1\) 条道路,并且任何两个村庄之间有且仅有一条路径可达.游戏开始时,玩家可以任意选择一个村庄,瞬 ...
- 洛谷 P3320 [SDOI2015]寻宝游戏
因为寻宝路径是一个环,所以寻宝花费的最小时间与起点无关.宝应当按照所有宝藏所在位置的 dfs 序进行才能够使得花费的时间最短.设 \(dist_i\) 表示 \(i\) 到树根的最短距离,那么树上任意 ...
- luogu P3320 [SDOI2015]寻宝游戏
大意:给定树, 要求维护一个集合, 支持增删点, 询问从集合中任取一点作为起点, 遍历完其他点后原路返回的最短长度. 集合中的点按$dfs$序排列后, 最短距离就为$dis(s_1,s_2)+...+ ...
- [BZOJ3991][SDOI2015]寻宝游戏
[BZOJ3991][SDOI2015]寻宝游戏 试题描述 小B最近正在玩一个寻宝游戏,这个游戏的地图中有N个村庄和N-1条道路,并且任何两个村庄之间有且仅有一条路径可达.游戏开始时,玩家可以任意选择 ...
- bzoj 3991: [SDOI2015]寻宝游戏 虚树 set
目录 题目链接 题解 代码 题目链接 bzoj 3991: [SDOI2015]寻宝游戏 题解 发现每次答案就是把虚树上的路径*2 接在同一关键点上的点的dfs序是相邻的 那么用set动态维护dfs序 ...
- 【LG3320】[SDOI2015]寻宝游戏
[LG3320][SDOI2015]寻宝游戏 题面 洛谷 题解 不需要建虚树的虚树2333... 贪心地想一下,起始节点肯定是在关键点上,访问顺序就是\(dfs\)序. 那么对于每次询问, \[ An ...
随机推荐
- Go单元测试注意事项及测试单个方法和整个文件的命令
Go程序开发过程中免不了要对所写的单个业务方法进行单元测试,Go提供了 "testing" 包可以实现单元测试用例的编写,不过想要正确编写单元测试需要注意以下三点: Go文件名必须 ...
- users命令详解
基础命令学习目录 原文链接:https://blog.csdn.net/m0_38132420/article/details/78861464 users命令用于显示当前登录系统所有的用户的用户列表 ...
- iOS 模块化、组件化方案探索(利用cocoapods 、git 创建私有仓库)
来自bang's blog http://blog.cnbang.net/tech/3080/ 模块化 简单来说,模块化就是将一个程序按照其功能做拆分,分成相互独立的模块,以便于每个模块只包含与其功能 ...
- 如何知道一个App的包名呢
包名(Package name)是Android系统中判断一个APP的唯一标识 记录我获取包名的几种方式 方法一:通过cmd命令,打开你要获取包名的APP 1.adb shell 2.dumpsys ...
- WebGL学习笔记二
前一章就是第二章主要学的是通过WebGL实现先是在webGL内赋值,但是不实用后来通过定义attribute和uniform存储限定符来将javascript中的数据传到webGL中,大致的流程是1. ...
- C++自学随笔(2)
引用 就像人的别名,人不能只有别名,变量也不能只有引用. 指针类型的引用:*&指针引用名 = 指针. 如int a = 10;int *p =&a;int *&q =p1 co ...
- asp.net登录验证FormsAuthenticationTicket和FormsAuthentication类
登录部分使用的类 FormsAuthentication 为 Web 应用程序管理 Forms 身份验证服务. 配置启用身份验证,WEB.config配置: <system.web> ...
- 【CSAPP笔记】2. 整型运算
现在想补补推荐这本书的理由. Most books on systems-computer architecture, compilers, operating systems, and networ ...
- Think In Java读书笔记:内部类覆盖及其初始化
本文相关章节:第十章 内部类 10.10 内部类可以被覆盖吗 在读至本节第二个范例代码时(及下方的代码),我对输出结果中的第一个“Egg.Yolk()”很不理解,为什么它会第一个地方输出. 我起初认为 ...
- Hibernate(十)
1.批处理 //批处理 :批量处理 //批量插入数据 @Test public void addData(){ Random random=new Random(); for(int i=1;i< ...