终于把这个坑填了..

按重心分治建树,每个点存两个堆,第一个存的是这个点子树中的点到父重心的距离,第二个存的是子节点第一个堆的堆顶,同时有一个全局答案堆,存的是每个点第二个堆的最大值+次大值。

20亿个蒟蒻们给了我一份漏洞百出的代码,害的我想了半天..

#include <cstdio>
#include <queue>
using namespace std; const int N=,M=;
char op[];
int n,e,x,y,q,tt,sm,mn,rt,hd[N],nx[M],to[M],v[N],f[N],a[N],d[N],sz[N],g[N][];
struct hep {
priority_queue<int> ad,dl;
void pu(int x) {ad.push(x);}
void po(int x) {dl.push(x);}
int tp() {while(!dl.empty()&&ad.top()==dl.top()) ad.pop(),dl.pop(); return ad.top();}
int gt() {int a=tp(); ad.pop(); int b=tp(); ad.push(a); return a+b;}
}a1,q1[N],q2[N];
void ad(int x,int y) {to[++e]=y,nx[e]=hd[x],hd[x]=e;} void dfs2(int x) {for(int i=hd[x];i;i=nx[i]) if(!d[to[i]]) d[to[i]]=d[x]+,g[to[i]][]=x,dfs2(to[i]);}
int lca(int x,int y) {
if(d[x]<d[y]) swap(x,y);
for(int i=;~i;i--) if(d[g[x][i]]>=d[y]) x=g[x][i];
if(x==y) return x;
for(int i=;~i;i--) if(g[x][i]^g[y][i]) x=g[x][i],y=g[y][i];
return g[x][];
}
int ds(int x,int y) {return d[x]+d[y]-d[lca(x,y)]*;}
void dfs(int x,int fa) {
sz[x]=;
for(int i=hd[x];i;i=nx[i]) if(to[i]!=fa&&!v[to[i]]) dfs(to[i],x),sz[x]+=sz[to[i]];
}
void gt(int x,int fa) {
int mx=; sz[x]=;
for(int i=hd[x];i;i=nx[i]) if(to[i]!=fa&&!v[to[i]]) gt(to[i],x),sz[x]+=sz[to[i]],mx=max(mx,sz[to[i]]);
mx=max(mx,tt-sz[x]);
if(mx<mn) mn=mx,rt=x;
}
void bd(int x,int y) {
gt(x,),f[rt]=y,v[rt]=,dfs(rt,); int k=rt;
for(int i=hd[rt];i;i=nx[i]) if(!v[to[i]]) tt=mn=sz[to[i]],bd(to[i],k);
}
void cal(int x) {
int u=x;
if(q2[x].ad.size()-q2[x].dl.size()>=) a1.po(q2[x].gt());
q2[x].pu();
if(q2[x].ad.size()-q2[x].dl.size()>=) a1.pu(q2[x].gt());
while(f[u]) {
if(q2[f[u]].ad.size()-q2[f[u]].dl.size()>=) a1.po(q2[f[u]].gt());
if(q1[u].ad.size()-q1[u].dl.size()) q2[f[u]].po(q1[u].tp());
q1[u].pu(ds(f[u],x)),q2[f[u]].pu(q1[u].tp());
if(q2[f[u]].ad.size()-q2[f[u]].dl.size()>=) a1.pu(q2[f[u]].gt());
u=f[u];
}
}
void cal2(int x) {
int u=x;
if(q2[x].ad.size()-q2[x].dl.size()>=) a1.po(q2[x].gt());
q2[x].po();
if(q2[x].ad.size()-q2[x].dl.size()>=) a1.pu(q2[x].gt());
while(f[u]) {
if(q2[f[u]].ad.size()-q2[f[u]].dl.size()>=) a1.po(q2[f[u]].gt());
q2[f[u]].po(q1[u].tp()),q1[u].po(ds(f[u],x));
if(q1[u].ad.size()-q1[u].dl.size()) q2[f[u]].pu(q1[u].tp());
if(q2[f[u]].ad.size()-q2[f[u]].dl.size()>=) a1.pu(q2[f[u]].gt());
u=f[u];
}
} int main() {
scanf("%d",&n),sm=n;
for(int i=;i<n;i++) scanf("%d%d",&x,&y),ad(x,y),ad(y,x);
tt=mn=n,bd(,),d[]=,dfs2();
for(int j=;j<;j++)
for(int i=;i<=n;i++)
g[i][j]=g[g[i][j-]][j-];
for(int i=;i<=n;i++) cal(i);
scanf("%d",&q);
while(q--) {
scanf("%s",op);
if(op[]=='G') {
if(sm<=) printf("%d\n",sm-);
else printf("%d\n",a1.tp());
}
else {
scanf("%d",&x),a[x]^=;
if(a[x]) sm--,cal2(x); else sm++,cal(x);
}
}
return ;
}

BZOJ1095(动态点分治+堆)的更多相关文章

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

    题目描述 捉迷藏 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiajia.Wind和孩子们决定在家里玩捉迷藏游戏.他们的家很大且构造很奇特,由N个屋子和N-1条双向走廊组成,这 ...

  2. BZOJ_1095_[ZJOI2007]Hide 捉迷藏_动态点分治+堆

    BZOJ_1095_[ZJOI2007]Hide 捉迷藏_动态点分治+堆 Description 捉迷藏 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiajia.Wind和孩子 ...

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

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

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

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

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

    [BZOJ1095][ZJOI2007]Hide 捉迷藏 Description 捉迷藏 Jiajia和Wind是一对恩爱的夫妻,并且他们有很多孩子.某天,Jiajia.Wind和孩子们决定在家里玩捉 ...

  6. bzoj 1095 Hide 捉迷藏 - 动态点分治 -堆

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

  7. BZOJ 1095: [ZJOI2007]Hide 捉迷藏 动态点分治+堆

    写了7k多,可以说是一己之力切掉了这道毒瘤题~ 开 $3$ 种堆,分别维护每个子树最大深度,以及每个节点在点分树中对父亲的贡献,和全局的最优解. 由于需要支持堆的删除,所以写起来特别恶心+麻烦. 细节 ...

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

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

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

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

随机推荐

  1. 织梦dedecms默认网站地图sitemap.html优化

    网站地图对于网站优化很重要,搜索引擎就是靠网站地图去收录网站页面,本文主要讲解优化织梦自带的网站地图功能.     织梦自带的网站地图使用方法:织梦后台--生成--HTML更新--更新网站地图,可以在 ...

  2. java截取一个字符串正数或倒数某个特定字符前后的内容

    取出正数第二个“.”后面的内容 public class TestCode { public static void main(String[] args) { String str ="2 ...

  3. 调用WCF时,调用已超过传入消息(65536)的最大消息大小配额。若要增加配额,请使用相应绑定。

    解决方案: 其实只要在客户端配置文件中加上如下紫色粗体属性( maxReceivedMessageSize): <?xml version="1.0" encoding=&q ...

  4. Xdebug的优点!php代码开发

    可以跟踪函数,知道那个函数出错,之前只是输出一点调试信息,只是哪一行错了,并且是白色 如果是死循环,debug会对死循环判断一百次的循环,并且会给出每一次循环的时间,内存,和函数名,位置.根据时间可以 ...

  5. Collaborative Filtering(协同过滤)算法详解

    基本思想 基于用户的协同过滤算法是通过用户的历史行为数据发现用户对商品或内容的喜欢(如商品购买,收藏,内容评论或分享),并对这些喜好进行度量和打分.根据不同用户对相同商品或内容的态度和偏好程度计算用户 ...

  6. ssh框架-Struts2(二)

    上篇文章我们了解了怎么配置struts.xml文件,以及前端控制器配置怎么配置,,Action进阶,Result结果配置,Struts2中的Servlet的API的访问,以及怎么获得请求参数.今天我们 ...

  7. Python内置函数(33)——any

    英文文档: any(iterable) Return True if any element of the iterable is true. If the iterable is empty, re ...

  8. Python内置函数(8)——bool

    英文文档: class bool([x]) Return a Boolean value, i.e. one of True or False. x is converted using the st ...

  9. WPF自定义控件与样式-自定义按钮(Button)

    一.前言 程序界面上的按钮多种多样,常用的就这几种:普通按钮.图标按钮.文字按钮.图片文字混合按钮.本文章记录了不同样式类型的按钮实现方法. 二.固定样式的按钮 固定样式的按钮一般在临时使用时或程序的 ...

  10. js回顾(DOM中标签的CRUD,表格等)

    01-DOM中的创建和添加标签 02-删除替换克隆标签 03-全选全不选反选 04-新闻字体 05-表格增删 06-动态生成表格 07-表格隔行变色 08-左到右右到左(将左边的标签移动到右边) 09 ...