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]寻宝游戏 解题报告的更多相关文章

  1. P3320 [SDOI2015]寻宝游戏

    题目 P3320 [SDOI2015]寻宝游戏 做法 很巧妙的一种思路,懂了之后觉得大水题 首先要知道:在一棵树上标记一些点,然后从任意一点出发,遍历所有的的最小路径为\(dfs\)序从小到大遍历 那 ...

  2. Luogu P3320 [SDOI2015]寻宝游戏 / 异象石 【LCA/set】

    期末考试结束祭! 在期末考试前最后一发的测试中,异象石作为第二道题目出现QAQ.虽然知道是LCA图论,但还是敲不出来QAQ. 花了两天竞赛课的时间搞懂(逃 异象石(stone.pas/c/cpp)题目 ...

  3. [洛谷P3320] SDOI2015 寻宝游戏

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

  4. [bzoj3991] [洛谷P3320] [SDOI2015] 寻宝游戏

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

  5. 洛谷 P3320 [SDOI2015]寻宝游戏

    因为寻宝路径是一个环,所以寻宝花费的最小时间与起点无关.宝应当按照所有宝藏所在位置的 dfs 序进行才能够使得花费的时间最短.设 \(dist_i\) 表示 \(i\) 到树根的最短距离,那么树上任意 ...

  6. luogu P3320 [SDOI2015]寻宝游戏

    大意:给定树, 要求维护一个集合, 支持增删点, 询问从集合中任取一点作为起点, 遍历完其他点后原路返回的最短长度. 集合中的点按$dfs$序排列后, 最短距离就为$dis(s_1,s_2)+...+ ...

  7. [BZOJ3991][SDOI2015]寻宝游戏

    [BZOJ3991][SDOI2015]寻宝游戏 试题描述 小B最近正在玩一个寻宝游戏,这个游戏的地图中有N个村庄和N-1条道路,并且任何两个村庄之间有且仅有一条路径可达.游戏开始时,玩家可以任意选择 ...

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

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

  9. 【LG3320】[SDOI2015]寻宝游戏

    [LG3320][SDOI2015]寻宝游戏 题面 洛谷 题解 不需要建虚树的虚树2333... 贪心地想一下,起始节点肯定是在关键点上,访问顺序就是\(dfs\)序. 那么对于每次询问, \[ An ...

随机推荐

  1. 关于spring boot 使用 mybatis plus INSERT的时候id报错

    mybatis plus 在INSERT的时候会默认自动设置插入id 我当时数据库采用的id自增. 在使用插入语句的时候并没有set  ID 但是它默认给了一大串 更改mybatis plus全局配置 ...

  2. 基于python的scrapy框架爬取豆瓣电影及其可视化

    1.Scrapy框架介绍 主要介绍,spiders,engine,scheduler,downloader,Item pipeline scrapy常见命令如下: 对应在scrapy文件中有,自己增加 ...

  3. Python 3 利用 Dlib 实现摄像头实时人脸检测和平铺显示

    1. 引言 在某些场景下,我们不仅需要进行实时人脸检测追踪,还要进行再加工:这里进行摄像头实时人脸检测,并对于实时检测的人脸进行初步提取: 单个/多个人脸检测,并依次在摄像头窗口,实时平铺显示检测到的 ...

  4. ats编译中增加透明度 选项

    在大多数情况下,如果环境支持透明度,则configure将自动启用它.对于其他环境,可能需要 配置configure 选项. --enable-posix-cap 这实现了POSIX功能,这是透明度所 ...

  5. 【视频编解码·学习笔记】13. 提取PPS信息程序

    PPS结构解析 与之前解析SPS方式类似 一.定义PPS类: 在3.NAL Unit目录下,新建PicParamSet.cpp和PicParamSet.h,在这两个文件中写入类的定义和函数实现. 类定 ...

  6. NDK 链接第三方静态库的方法

    将NDK编译的第三方静态拷贝到JNI目录下,在Android.mk中添加如下代码 以openssl静态库(libcrypto-static.a)为例 第一种链接方法:LOCAL_LDFLAGS := ...

  7. dos2unix命令详解

    基础命令学习目录首页 原文链接:https://blog.csdn.net/leedaning/article/details/53024290 使用git 的时候碰到git将unix换行符转换为wi ...

  8. vi/vim命令详解

    基础命令学习目录首页 原文链接:https://www.cnblogs.com/mondol/p/vi-examples.html 进入vi vi filename # 打开或新建文件,并将光标置于第 ...

  9. Python爬虫框架Scrapy学习笔记原创

     字号 scrapy [TOC] 开始 scrapy安装 首先手动安装windows版本的Twisted https://www.lfd.uci.edu/~gohlke/pythonlibs/#twi ...

  10. final文案+美工展示

    作业要求:https://edu.cnblogs.com/campus/nenu/SWE2017FALL/homework/1438 团队介绍:thunder 组成员及各位博客地址: 1.王航:htt ...