前言

好恶心的一道题,代码写了2.5h,调试调了5h+,局部重构了n遍。

题意

一棵树上的节点有黑白两色,初始为黑,支持修改颜色,查询最远黑点对。$n<=10^5,m<=5*10^5$

题解

ver 1

先考虑查询,可以在每个点保存一个堆s1存储子树内到这个点的路径。

再维护一个全局堆,把每个点的【最大和次大的值的和】放入堆内。

查询时直接取出堆顶。

ver 2

然后你发现你遇到了这种情况(1-2的路径被重复走了)

于是在每个点再开一个新的堆s2储存它的子数内到父亲的路径。

每个点的s1从儿子的s2.top()中取得。

这样每个点只会对父亲做一次贡献(这好像是一个常见的套路)

要修改的话就从这个点开始一直往父亲走,在走的时候顺便更新(具体看代码中的on和off)

ver 3

然后你发现你的代码时空复杂度都是$O(n^2)$的

于是用点分治重构树,这样树高就是$O(logn)$,空间复杂度均摊就是$O(n*logn)$的了。

这样求距离就要用lca了,时间复杂度$O(n*logn^2)$。

注意一定要用堆,平衡树会超时(我在这调了2h+)

这里用到了一个小技巧:可删除任意元素的堆。

具体就是对于每个堆再维护一个堆用于存储要删除的节点。

在取top的时候看看top是否与删除堆的top相等,如相等,则删除这个top。

代码

//#pragma comment(linker, "/STACK:268435456,268435456")
#include <iostream>
#include <cstdio>
#include <vector>
#include <queue>
using namespace std;
#define N 100010
#define int long long
class _set
{
public:
priority_queue<int> source,del;
int top()
{
while(!del.empty()&&source.top()==del.top()) source.pop(),del.pop();
return source.top();
}
int size()
{
return source.size()-del.size();
}
int sec()
{
int t=top(),ret;
del.push(t);
ret=top();
source.push(t);
return ret;
}
}s1[N],s2[N],ans;
#define _erase del.push
#define _push source.push
int to[N*2],nxt[N*2],head[N],cnt;
void connect(int a,int b)
{
for(int i=1;i<=2;i++)
{
to[++cnt]=b;
nxt[cnt]=head[a];
head[a]=cnt;
swap(a,b);
}
}
int sz[N],root,used[N],fa[N],tot,up[N][21],dep[N];
void dfs(int id,int from)
{
//printf("%d\n",id);
up[id][0]=from;
dep[id]=dep[from]+1;
for(int i=1;i<=20;i++) up[id][i]=up[up[id][i-1]][i-1];
for(int i=head[id];i;i=nxt[i])
{
if(to[i]==from) continue;
dfs(to[i],id);
}
}
int get_lca(int x,int y)
{
if(dep[x]<dep[y]) swap(x,y);
for(int i=20;dep[x]>dep[y];i--)
{
if(dep[up[x][i]] >= dep[y]) x=up[x][i];
}
if(x==y) return x;
for(int i=20;up[x][0]!=up[y][0];i--)
{
if(up[x][i]!=up[y][i])
{
x=up[x][i];
y=up[y][i];
}
}
return up[x][0];
}
int get_dis(int x,int y)
{
int lca=get_lca(x,y);
return dep[x]+dep[y]-2*dep[lca];
}
void del(_set &s)
{
ans._erase(s.top()+s.sec());
}
void add(_set &s)
{
ans._push(s.top()+s.sec());
}
void get_root(int id,int from,int size)
{
sz[id]=1;
int maxn=0;
for(int i=head[id];i;i=nxt[i])
{
if(to[i]==from||used[to[i]]) continue;
get_root(to[i],id,size);
sz[id]+=sz[to[i]];
maxn=max(maxn,sz[to[i]]);
}
if(max(maxn,size-sz[id])<=size/2)
root=id;
}
void get_size(int id,int from)
{
sz[id]=1;
for(int i=head[id];i;i=nxt[i])
{
if(to[i]==from||used[to[i]]) continue;
get_size(to[i],id);
sz[id]+=sz[to[i]];
}
}
void divide(int id)
{
used[id]=true;
for(int i=head[id];i;i=nxt[i])
{
if(used[to[i]]) continue;
//puts("$");
get_root(to[i],id,sz[to[i]]);
fa[root]=id;
get_size(root,0);
divide(root);
}
}
void off(int id)
{
s1[id].source.push(0);//允许路径在这里停下
if(s1[id].size()==2) add(s1[id]);//如果之前没有添加到答案堆,则添加
for(int t=id;fa[t];t=fa[t])//一路往上更新父亲以及答案堆
{
if(s1[fa[t]].size()>=2) del(s1[fa[t]]);//将答案堆中fa[t]旧的数据删除
if(s2[t].size()) s1[fa[t]].del.push(s2[t].top()); //将s1[fa[t]]中s2[t]旧的数据删除
s2[t]._push(get_dis(id,fa[t]));//因为现在路径可以从id开始,所以添加一条id->fa[t]的路径
s1[fa[t]]._push(s2[t].top());//将新的数据加入
if(s1[fa[t]].size()>=2) add(s1[fa[t]]);//同上
}
}
void on(int id)//基本同上
{
if(s1[id].size()==2) del(s1[id]);
s1[id]._erase(0);
for(int t=id;fa[t];t=fa[t])
{
if(s1[fa[t]].size()>=2) del(s1[fa[t]]);
if(s2[t].size()) s1[fa[t]]._erase(s2[t].top());
s2[t]._erase(get_dis(id,fa[t]));
if(s2[t].size()) s1[fa[t]]._push(s2[t].top());
if(s1[fa[t]].size()>=2) add(s1[fa[t]]);
}
}
int sta[N];
signed main()
{
int n,q;
cin>>n;
tot=n;
for(int i=1;i<n;i++)
{
int a,b;
scanf("%d%d",&a,&b);
connect(a,b);
}
dfs(1,0);
get_root(1,0,n);
get_size(root,0);
divide(root);
cin>>q;
for(int i=1;i<=n;i++) off(i);
for(int i=1;i<=q;i++)
{
char opt;
scanf(" %c",&opt);
int p;
if(opt=='G')
{
if(tot>=2) printf("%d\n", ans.top());
else if(tot==1) puts("0");
else puts("-1");
}
else
{
scanf("%d",&p);
if(sta[p]) off(p),tot++;
else on(p),tot--;
sta[p]^=1;
}
}
}

  

【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—捉迷藏

    http://www.lydsy.com/JudgeOnline/problem.php?id=1095 (题目链接) 题意 一棵树,求最远的两黑点之间的距离,每次可以将黑点染白或者将白点染黑. So ...

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

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

随机推荐

  1. Intellij IDEA 中的 Debug 控制台输出窗口不见了的解决办法

    在 Debug 工具窗口,如图点击左侧重置布局图标,你的console窗口就还原了.

  2. 复习:C语言基础知识1

    占位符: %d, %i,代表整数,%f-浮点,%s,字符串,%c,char. %p 指针,%fL 长long,%e科学计数,%g 小数或科学计数. C语言中的格式占位符: %a,%A 读入一个浮点值( ...

  3. Codeforces 878 E. Numbers on the blackboard

    Codeforces 878 E. Numbers on the blackboard 解题思路 有一种最优策略是每次选择最后面一个大于等于 \(0\) 的元素进行合并,这样做完以后相当于给这个元素乘 ...

  4. LOJ2392 JOISC2017 烟花棒 二分、贪心

    传送门 先二分一个最大速度\(v\). 分析移动的性质.很显然的事情是在火焰两边的所有人都会往火焰的方向以最快的速度运动,这样可以使当前位置更早获得火焰,同时当前拥有火焰的若干个人为了传递火焰自然也会 ...

  5. GoF的23种设计模式之结构型模式的特点和分类

    结构型模式描述如何将类或对象按某种布局组成更大的结构.它分为类结构型模式和对象结构型模式,前者采用继承机制来组织接口和类,后者釆用组合或聚合来组合对象. 由于组合关系或聚合关系比继承关系耦合度低,满足 ...

  6. 【华为云实战开发】10.经典的C++项目怎么在云端开发?【华为云技术分享】

    1 概述 1.1 文章目的 本文主要想为研发C++项目的企业或个人提供上云指导,通过本文中的示例项目 “音频解析器”,为开发者提供包括项目管理,代码托管,代码检查,编译构建,测试管理的操作指导,覆盖软 ...

  7. JS实现文件自动上传

    JS引用: <script type="text/javascript" src="~/bootstrap/js/fileinput.min.js"> ...

  8. AnimationClip压缩-动画文件压缩

    动画压缩方法一.常用方法1. Rig->Animation Type:改为Generic2. Animations->Anim.Compression:Optimal二.高级方法1. 去掉 ...

  9. Git教程-安装与创建版本库

    Git是一个分布式版本控制系统,他通过命令行使用的工具,Github是提供Git仓库托管服务的网站 安装参考: https://www.liaoxuefeng.com/wiki/89604348802 ...

  10. IntelliJ IDEA重命名变量的问题

    当我尝试使用Shift+ F6或简单地使用Refactor => Rename重命名变量时,有时intellij不仅重命名我想要的那个,而且还重命名具有相同名称的所有其他变量(在其他文件中)以及 ...