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 ...
随机推荐
- Centos 7 安装Zabbix
一.环境准备与说明: 1.zabbix server 版本:3.4.12 ,https://www.zabbix.com/download 2.zabbix agent版本:3.4.14,https: ...
- bitcoin PoW原理及区块创建过程
bitcoin PoW原理及区块创建过程 PoW 为了在点对点的基础上实现一个分布式时间戳服务器,我们需要使用PoW(Proof of Work)系统来达成共识.PoW过程就是寻找一个目标值的过程,当 ...
- GO/GOLANG程序员笔记大全
---------------------------------------- go 并发 // 注解:go 语言天生为程序并发所设计,可以说go的强项就是在cpu并发上的处理. // go 语言层 ...
- chattr和lsattr命令详解
基础命令学习目录首页 原文链接:http://www.ha97.com/5172.html PS:有时候你发现用root权限都不能修改某个文件,大部分原因是曾经用chattr命令锁定该文件了.chat ...
- rest_framework_api规范
目录 一.什么是RESTful 二.什么是API 三.RESTful API规范 四.基于Django实现API 五.基于Django Rest Framework框架实现 一. 什么是RESTful ...
- Notes of Daily Scrum Meeting(11.19)
Notes of Daily Scrum Meeting(11.19) 现在工程项目进入尾声了,我们的项目中还有一些问题需要解决,调试修改起来进度比较慢,所以昨天就没有贴出项目 进度,今天的团队工作总 ...
- 实验三 Java猜数字游戏开发
课程:Java实验 班级:201352 姓名:程涵 学号:20135210 成绩: 指导教师:娄佳鹏 实验日期:15.06.03 实验密级: ...
- 软件团队项目第一次Sprint评价(评价人:张家军)
组号 组名 缺点及建议 1 理财猫 (1)没有附带的计算器 (2)支入支出没有详细菜单说明 (3)界面背景单调 ...
- BeanUtils解决日期问题
ConvertUtils.register(new DateLocaleConverter(), Date.class); BeanUtils.populate(user, request.getPa ...
- 修复webpack自动刷新页面慢的问题
新建.babelrc文件,配置如下 { "presets": [ "es2015" ], "ignore":[ "react-ro ...