动态树分治,用三个set分别维护每个重心到每一个子树的距离种类、每个重心所有子树的最大值和次大值、全局答案的最大值。复杂度O(nlogn^2)

  代码

 #include<cstdio>
#include<algorithm>
#include<vector>
#include<set>
#define pb push_back
using namespace std;
const int N = ;
int size[N],flag[N],n,a,b,i,f[N],cnt,typ[N],deep[N];
int jump[N][];
int q;
char ch[];
vector<int> e[N];
multiset<int> dist[N],ans[N],Ans;
multiset<int>::iterator it;
int getroot(int x,int fa,int n)
{
int i,tmp=,rt=;
size[x]=;
for (i=;i<e[x].size();i++)
if ((e[x][i]!=fa)&&(!flag[e[x][i]]))
{
rt|=getroot(e[x][i],x,n);
size[x]+=size[e[x][i]];
if (size[e[x][i]]>n/) tmp=;
}
if (n-size[x]>n/) tmp=;
if (tmp==) return rt;else return x;
}
void gao(int x,int fa,int y,int dis)
{
dis++;
int i;
dist[y].insert(dis);
for (i=;i<e[x].size();i++)
if ((!flag[e[x][i]])&&(e[x][i]!=fa))
gao(e[x][i],x,y,dis);
}
void DFS(int x,int fa)
{
int i;
deep[x]=deep[fa]+;
jump[x][]=fa;
for (i=;i<=;i++)
jump[x][i]=jump[jump[x][i-]][i-];
for (i=;i<e[x].size();i++)
if (e[x][i]!=fa) DFS(e[x][i],x);
}
int lca(int a,int b)
{
int i;
if (deep[a]<deep[b]) a^=b^=a^=b;
for (i=;i>=;i--)
if (deep[jump[a][i]]>=deep[b]) a=jump[a][i];
if (a==b) return a;
for (i=;i>=;i--)
if (jump[a][i]!=jump[b][i]) a=jump[a][i],b=jump[b][i];
return jump[a][];
}
int getdis(int x,int y)
{
int z=lca(x,y);
return deep[x]+deep[y]-*deep[z];
}
void del(int x)
{
int tmp=;
if (ans[x].size()>=)
{
it=ans[x].end();
--it;tmp+=*it;
--it;tmp+=*it;
it=Ans.lower_bound(tmp);
Ans.erase(it);
}
}
void ins(int x)
{
int tmp=;
if (ans[x].size()>=)
{
it=ans[x].end();
--it;tmp+=*it;
--it;tmp+=*it;
Ans.insert(tmp);
}
}
void DEL(int y)
{
if (dist[y].size())
{
it=dist[y].end();--it;
int tmp=*it;
it=ans[f[y]].lower_bound(tmp);
ans[f[y]].erase(it);
}
}
void INS(int y)
{
if (dist[y].size())
{
it=dist[y].end();--it;
int tmp=*it;
ans[f[y]].insert(tmp);
}
}
int change(int x,int y)
{
del(x);
if (typ[x]==)
{
it=ans[x].lower_bound();
ans[x].erase(it);
}
else
ans[x].insert();
while (x)
{
ins(x);
if (f[x])
{
del(f[x]);
DEL(x);
int dis=getdis(f[x],y);
if (typ[y]==)
{
it=dist[x].lower_bound(dis);
dist[x].erase(it);
}
else
dist[x].insert(dis);
INS(x);
}
x=f[x];
}
}
int dfs(int x,int n,int fa)
{
int i,y;
x=getroot(x,,n);
f[x]=fa;
flag[x]=;
ans[x].insert();
for (i=;i<e[x].size();i++)
if (!flag[e[x][i]])
{
if (size[e[x][i]]>size[x])
y=dfs(e[x][i],n-size[x],x);
else
y=dfs(e[x][i],size[e[x][i]],x);
gao(e[x][i],,y,);
it=dist[y].end();
ans[x].insert(*(--it));
}
ins(x);
flag[x]=;
return x;
}
int main()
{
scanf("%d",&n);
for (i=;i<n;i++)
{
scanf("%d%d",&a,&b);
e[a].pb(b);
e[b].pb(a);
}
dfs(,n,);
DFS(,);
scanf("%d",&q);
for (i=;i<=q;i++)
{
scanf("%s",ch+);
if (ch[]=='G')
{
if (cnt==n)
printf("-1\n");
else
if (cnt==n-)
printf("0\n");
else
{
it=Ans.end();--it;
printf("%d\n",*it);
}
}
else
if (ch[]=='C')
{
scanf("%d",&a);
if (typ[a]==)
cnt++,typ[a]=;
else
cnt--,typ[a]=;
change(a,a);
}
}
}

BZOJ1095 [ZJOI2007]Hide 捉迷藏的更多相关文章

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

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

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

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

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

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

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

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

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

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

  6. bzoj千题计划245:bzoj1095: [ZJOI2007]Hide 捉迷藏

    http://www.lydsy.com/JudgeOnline/problem.php?id=1095 查询最远点对,带修改 显然可以用动态点分治 对于每个点,维护两个堆 堆q1[x] 维护 点分树 ...

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

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

  8. BZOJ1095 [ZJOI2007]Hide 捉迷藏 【动态点分治 + 堆】

    题目链接 BZOJ1095 题解 传说中的动态点分治,一直不敢碰 今日一会,感觉其实并不艰涩难懂 考虑没有修改,如果不用树形dp的话,就得点分治 对于每个重心,我们会考虑其分治的子树内所有点到它的距离 ...

  9. bzoj1095: [ZJOI2007]Hide 捉迷藏 线段树维护括号序列 点分治 链分治

    这题真是十分难写啊 不管是点分治还是括号序列都有一堆细节.. 点分治:时空复杂度$O(n\log^2n)$,常数巨大 主要就是3个堆的初始状态 C堆:每个节点一个,为子树中的点到它父亲的距离的堆. B ...

随机推荐

  1. C#怎样通过url调用接口

    在做一些项目过程中,我们常常总避免不了要调用接口,那么怎么通过url调用借口呢.我今天浅显的写一下. 首先要获取你访问链接的App Key 和 App Secret 那么什么是App Key 和 Ap ...

  2. jquery实现限制textarea输入字数

    ie中可用onpropertychange监听改变事件 火狐和谷歌可用oninput监听改变事件 综合: //使系统中class='text-length'的输入框只能输入200字符(主要用于text ...

  3. python实现之决策树

    一.Predict survival on the Titanic 使用泰坦尼克号上的乘客数据,对乘客是否存活进行预测 1.观察数据集合 可能遇到的问题 训练集和测试集特征值得属性并不重合.连续属性和 ...

  4. Html简单介绍

    一.Html 1.万维网的核心语言,高大上称之为超文本标记语言(Html)的第五次修改 2.完成的时间:2014年10月29日 3.我们需要了解有一定高度的知识: WHATWG  WEB超文本应用技术 ...

  5. Unity学习疑问记录之查找

    unity中提供了获取对象的五种方法: 通过对象名称(Find方法) 通过标签获取单个游戏对象(FindWithTag方法) 通过标签获取多个游戏对象(FindGameObjectsWithTags方 ...

  6. 清理session的案例

    and OPNAME ='Sort Output'; and OPNAME ='Sort Output'; and OPNAME ='Sort Output' ); select 'alter sys ...

  7. PHP 使用 debug_print_backtrace() 或 debug_backtrace() 打印栈轨迹

    <?php /* 使用debug_print_backtrace() 或 debug_backtrace() 打印栈轨迹 */ function fun1() { print "Hel ...

  8. 微信小程序 不在以下合法域名列表中,请参考文档:https://mp.weixin.qq.com/debug/wxadoc/dev/api/network-request.html

    微信小程序  不在以下合法域名列表中,请参考文档:https://mp.weixin.qq.com/debug/wxadoc/dev/api/network-request.html 友情提示: 大家 ...

  9. TabSiPlus发布1749版本

    拖了几天,终于可以发布了,下载地址如下: http://www.winmsg.com/download/TabSiPlus_0_99b2_1749.rar 这个包里没有附带VC的运行库,考虑到大多数人 ...

  10. @Html.Raw()

    在用VS 2015写代码时,匹配邮箱的正则表达式 /^\w+([-+.]\w+)*@\w+([-.]\w+)*\.\w+([-.]\w+)*$/​ @报错  主要因为 @是MVC里的关键字,所以不能直 ...