http://www.lydsy.com/JudgeOnline/problem.php?id=1095 (题目链接)

题意

  一棵树,求最远的两黑点之间的距离,每次可以将黑点染白或者将白点染黑。

Solution

  动态树分治,%%%重庆省选AK爷

  点分治的过程是对树块找重心之后分成多个小树块,降低规模分别处理的过程,把链的信息收到其中“最高重心”上,从所有的重心处像分治中的不同子树索取到重心的链,就可以覆盖所有链的信息。动态点分治就像把序列分治变成线段树一样,在分治的架子上加了信息维护,实现树链信息维护与查询。

  需要什么?

  每个重心需要其每个分离子树到它的信息(很重要,否则形成链的重复部分,并且还需要一个自己到自己的空信息维护单链上来的信息)

  每个重心需要它到父分治块的信息

  全局需要维护每个重心的信息

  因此,考虑问题的静态版本,需要维护每个节点每个子树内的最长链,那么为了修改,最大化应换成堆维护。

  每个重心维护一个堆,表示其分离子树中每个到它的最长链

  每个重心维护一个堆,表示其块内到父重心的最长链

  全局维护一个堆,表示每个重心处的最长链

  修改时只要从一个节点作为重心的块开始向上修改,就可以遍历到所有包含它的块,修改到父重心的信息,修改父重心的信息

  注意开始时的节点的分离子树那个堆要插一个0表示单链,而改掉节点值的时候0的存在性也要相应地变化。

细节

  今天我机子真是出了鬼了,几个明显会WA的错误竟然拍不出来,搞得我都怀疑自己存在的价值了→_→(虽然我也不知道这二者之间有什么联系)

代码

// bzoj1095
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<queue>
#include<ctime>
#define LL long long
#define inf 1ll<<30
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std; const int maxn=100010;
int light[maxn],bin[30],par[maxn],head[maxn];
int n,Q; struct edge {int to,next;}e[maxn<<1];
struct hope {
priority_queue<int> q,del;
int top() {
while (!del.empty() && q.top()==del.top()) q.pop(),del.pop();
return q.top();
}
void pop() {
while (!del.empty() && q.top()==del.top()) q.pop(),del.pop();
q.pop();
}
int size() {
return q.size()-del.size();
}
void push(int x) {
q.push(x);
}
void erase(int x) {
del.push(x);
}
}q[maxn],d[maxn],ans; namespace LittleTrick {
int cnt,deep[maxn],fa[maxn][30]; inline void link(int u,int v) {
e[++cnt]=(edge){v,head[u]};head[u]=cnt;
e[++cnt]=(edge){u,head[v]};head[v]=cnt;
}
inline void dfs(int x) {
for (int i=1;i<=20;i++) fa[x][i]=fa[fa[x][i-1]][i-1];
for (int i=head[x];i;i=e[i].next) if (e[i].to!=fa[x][0]) {
deep[e[i].to]=deep[x]+1;
fa[e[i].to][0]=x;
dfs(e[i].to);
}
}
inline int lca(int x,int y) {
if (deep[x]<deep[y]) swap(x,y);
int t=deep[x]-deep[y];
for (int i=0;bin[i]<=t;i++) if (bin[i]&t) x=fa[x][i];
if (x==y) return x;
for (int i=20;i>=0;i--) if (fa[x][i]!=fa[y][i]) x=fa[x][i],y=fa[y][i];
return fa[x][0];
}
inline void insert(hope &a) {
if (a.size()>1) {
int tmp=a.top();a.pop();
ans.push(tmp+a.top());a.push(tmp);
}
}
inline void erase(hope &a) {
if (a.size()>1) {
int tmp=a.top();a.pop();
ans.erase(tmp+a.top());a.push(tmp);
}
}
inline int dis(int x,int y) {
return deep[x]+deep[y]-2*deep[lca(x,y)];
}
}
using namespace LittleTrick; namespace NodeDivide {
int Dargen,sum;
int size[maxn],f[maxn],vis[maxn]; inline void caldargen(int x,int fa) {
f[x]=0;size[x]=1;
for (int i=head[x];i;i=e[i].next) if (e[i].to!=fa && !vis[e[i].to]) {
caldargen(e[i].to,x);
size[x]+=size[e[i].to];
f[x]=max(f[x],size[e[i].to]);
}
f[x]=max(f[x],sum-size[x]);
if (f[x]<f[Dargen]) Dargen=x;
}
inline void caldeep(int x,int fa,int p) {
d[p].push(dis(x,par[p]));
for (int i=head[x];i;i=e[i].next)
if (e[i].to!=fa && !vis[e[i].to]) caldeep(e[i].to,x,p);
}
inline void work(int x) {
vis[x]=1;
q[x].push(0);
caldeep(x,0,x);
for (int i=head[x];i;i=e[i].next) if (!vis[e[i].to]) {
sum=size[e[i].to];Dargen=0;
caldargen(e[i].to,x);
par[Dargen]=x;// par[e[i].to]=x;
work(Dargen);//work(e[i].to);
//q[x].push(d[e[i].to].top());
}
q[par[x]].push(d[x].top());
insert(q[x]);
}
inline void Init() {
f[Dargen=0]=inf;sum=n;
caldargen(1,0);
work(Dargen);
}
}
using namespace NodeDivide; namespace Query {
inline void On(int x) {
erase(q[x]);
q[x].erase(0);
insert(q[x]);
for (int i=x;par[i];i=par[i]) {
erase(q[par[i]]);
if (d[i].size()) q[par[i]].erase(d[i].top());
d[i].erase(dis(par[i],x));
if (d[i].size()) q[par[i]].push(d[i].top());
insert(q[par[i]]);
}
}
inline void Off(int x) {
erase(q[x]);
q[x].push(0);
insert(q[x]);
for (int i=x;par[i];i=par[i]) {
erase(q[par[i]]);
if (d[i].size()) q[par[i]].erase(d[i].top());
d[i].push(dis(par[i],x));
if (d[i].size()) q[par[i]].push(d[i].top());
insert(q[par[i]]);
}
}
}
using namespace Query; int main() {
bin[0]=1;for (int i=1;i<=20;i++) bin[i]=bin[i-1]<<1;
scanf("%d",&n);
for (int u,v,i=1;i<n;i++) {
scanf("%d%d",&u,&v);
link(u,v);
}
dfs(1);
Init();
scanf("%d",&Q);
char ch[5];
for (int cnt=n,x,i=1;i<=Q;i++) {
scanf("%s",ch);
if (ch[0]=='G') {
if (cnt<=1) printf("%d\n",cnt-1);
else printf("%d\n",ans.top());
}
else {
scanf("%d",&x);
if (!light[x]) On(x),light[x]=!light[x],cnt--;
else Off(x),light[x]=!light[x],cnt++;
}
}
return 0;
}

  

【bzoj1095】 ZJOI2007—捉迷藏的更多相关文章

  1. 动态点分治:Bzoj1095: [ZJOI2007]Hide 捉迷藏

    简介 这是我自己的一点理解,可能写的不好 点分治都学过吧.. 点分治每次找重心把树重新按重心的深度重建成了一棵新的树,称为分治树 这个树最多有log层... 动态点分治:记录下每个重心的上一层重心,这 ...

  2. 【BZOJ1095】捉迷藏(动态点分治)

    [BZOJ1095]捉迷藏(动态点分治) 题面 BZOJ 题解 动态点分治板子题 假设,不考虑动态点分治 我们来想怎么打暴力: \(O(n)DP\)求树的最长链 一定都会.不想解释了 所以,利用上面的 ...

  3. 洛谷 P2056 [ZJOI2007]捉迷藏 解题报告

    P2056 [ZJOI2007]捉迷藏 题目描述 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiajia.Wind和孩子们决定在家里玩捉迷藏游戏.他们的家很大且构造很奇特,由\ ...

  4. 树上最长链 Farthest Nodes in a Tree LightOJ - 1094 && [ZJOI2007]捉迷藏 && 最长链

    树上最远点对(树的直径) 做法1:树形dp 最长路一定是经过树上的某一个节点的. 因此: an1[i],an2[i]分别表示一个点向下的最长链和次长链,次长链不存在就设为0:这两者很容易求 an3[i ...

  5. [bzoj1095][ZJOI2007]Hide 捉迷藏 点分树,动态点分治

    [bzoj1095][ZJOI2007]Hide 捉迷藏 2015年4月20日7,8876 Description 捉迷藏 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiaji ...

  6. BZOJ1095:[ZJOI2007]Hide 捉迷藏(动态点分治)

    Description 捉迷藏 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiajia.Wind和孩子们决定在家里玩 捉迷藏游戏.他们的家很大且构造很奇特,由N个屋子和N-1条 ...

  7. BZOJ1095 [ZJOI2007]Hide 捉迷藏 动态点分治 堆

    原文链接https://www.cnblogs.com/zhouzhendong/p/BZOJ1095.html 题目传送门 - BZOJ1095 题意 有 N 个点,每一个点是黑色或者白色,一开始所 ...

  8. BZOJ1095: [ZJOI2007]Hide 捉迷藏【线段树维护括号序列】【思维好题】

    Description 捉迷藏 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiajia.Wind和孩子们决定在家里玩 捉迷藏游戏.他们的家很大且构造很奇特,由N个屋子和N-1条 ...

  9. BZOJ1095: [ZJOI2007]Hide 捉迷藏【动态点分治】

    Description 捉迷藏 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiajia.Wind和孩子们决定在家里玩 捉迷藏游戏.他们的家很大且构造很奇特,由N个屋子和N-1条 ...

随机推荐

  1. 2018爆零记第二弹之day0

    话说初赛水了个70分,ε=(´ο`*)))唉,还是太菜了. 今天两点左右到了电子科大对面宾馆,收拾安顿好后又去电子科大踩点. 进门又走过了不长不短的水杉道,来到了不大不小的西湖(为什么是这个名字... ...

  2. Netty源码分析第5章(ByteBuf)---->第10节: SocketChannel读取数据过程

    Netty源码分析第五章: ByteBuf 第十节: SocketChannel读取数据过程 我们第三章分析过客户端接入的流程, 这一小节带大家剖析客户端发送数据, Server读取数据的流程: 首先 ...

  3. python-gevent模块实现socket大并发

    服务器端:gevent_server.py 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 ...

  4. Linux 学习记录 20170218

    一.Linux 硬件查看命令     ----/proc 文件系统是一种内核和内核模块用来向进程(process) 发送信息的机制.我们可以从这个文件里获取到系统的相关信息. 1.显卡信息dmesg ...

  5. 第二阶段每日站立会议Second Day

    昨天我在手机端安装cpp后进行界面效果测试以及进一步完善 今天对图片显示的大小进行调整 遇到的问题:当图片太小时,显示一块灰色区域,不美观

  6. mianshi

    https://blog.csdn.net/u012557610/article/details/80350099 https://blog.csdn.net/liuqiyao_01/article/ ...

  7. 手机连接wifi 访问本地服务器网站

    手机连本地wifi后访问 http://192.168.155.1:8001/loc 版权声明:本文为博主原创文章,未经博主允许不得转载.

  8. spring冲刺阶段之团队工作总结

    一.小组成员: 王俊凯(项目经理) 罗林杰(产品负责人) 王逸辉(Master) 罗凯杰 二.任务分配情况 王俊凯:生成题目的代码编写并提出编写意见 罗林杰:负责把按钮和界面内容连接到代码上及主要代码 ...

  9. Journal entry of the thirteenth chapter to chapter seventeenth(第十三章和十七章阅读与疑问)

    第十三章: 软件测试的意义在于: a.     发现软件错误: b.     有效定义和实现软件成分由低层到高层的组装过程: c.     验证软件是否满足任务书和系统定义文档所规定的技术要求: d. ...

  10. PTA计算平均值(一波三折)

    PTA计算平均值( 一波三折) 现在为若干组整数分别计算平均值. 已知这些整数的绝对值都小于100,每组整数的数量不少于1个,不大于20个. 输入格式:首先输入K(不小于2,不大于20).接下来每一行 ...