简介

这是我自己的一点理解,可能写的不好

点分治都学过吧。。

点分治每次找重心把树重新按重心的深度重建成了一棵新的树,称为分治树

这个树最多有log层。。。

动态点分治:记录下每个重心的上一层重心,这棵分治树就确定了

修改就暴力在分治树中向上改,反正是log的

至于为什么叫动态点分治我不知道。。。我觉得就是点分治

做题时最主要的难点不在点分治,在于维护什么和怎样维护

例题

Bzoj1095: [ZJOI2007]Hide 捉迷藏

先搞出这个分治树,然后基本和点分治无关了

以下基于分治树

每个重心开两个堆

第一个堆记录子树中所有节点到重心的距离

第二个堆记录所有子节点的第一个堆的堆顶

那么一个节点的第二个堆堆中的最大值和次大值加起来就是子树中经过这个节点的最长链

然后开一个全局的堆,记录所有第二个堆中最大值和次大值之和

堆顶就是答案

修改就分治树中暴跳重心,大力讨论一番

代码和思路来自hzwer

# include <bits/stdc++.h>
# define RG register
# define IL inline
# define Fill(a, b) memset(a, b, sizeof(a))
using namespace std;
typedef long long ll;
const int _(2e5 + 10); IL ll Read(){
RG ll x = 0, z = 1; RG char c = getchar();
for(; c < '0' || c > '9'; c = getchar()) z = c == '-' ? -1 : 1;
for(; c >= '0' && c <= '9'; c = getchar()) x = (x << 1) + (x << 3) + (c ^ 48);
return x * z;
} int n, nxt[_], to[_], fst[_], cnt, deep[_], size[_], frt[_], rt, mx[_], sz;
int eul[20][_], pw[20] = {1}, lg[_], id[_], tot;
bool cls[_], vis[_];
struct Heap{
priority_queue <int> A, B;
IL void Push(RG int x){ A.push(x); } IL void Del(RG int x){ B.push(x); } IL void Pop(){ while(!B.empty() && A.top() == B.top()) A.pop(), B.pop(); A.pop(); } IL int Top(){ while(!B.empty() && A.top() == B.top()) A.pop(), B.pop(); return !A.empty() ? A.top() : 0; } IL int Size(){ return A.size() - B.size(); } IL int _Top(){ if(Size() < 2) return 0; RG int x = Top(); Pop(); RG int y = Top(); Push(x); return y; }
} A, B[_], C[_]; IL void Add(RG int u, RG int v){ to[cnt] = v; nxt[cnt] = fst[u]; fst[u] = cnt++; }
/***********************************建立分治树***************************************/
IL void Getroot(RG int u, RG int ff){
size[u] = 1; mx[u] = 0;
for(RG int e = fst[u]; e != -1; e = nxt[e]){
if(to[e] == ff || vis[to[e]]) continue;
Getroot(to[e], u);
size[u] += size[to[e]];
mx[u] = max(mx[u], size[to[e]]);
}
mx[u] = max(mx[u], sz - size[u]);
if(mx[u] < mx[rt]) rt = u;
} IL void Create(RG int u, RG int ff){
frt[u] = ff; /*记录上层重心*/ vis[u] = 1;
for(RG int e = fst[u]; e != -1; e = nxt[e]){
if(vis[to[e]]) continue;
rt = 0; sz = size[to[e]];
Getroot(to[e], 0);
Create(rt, u);
}
}
/***********************************************************************************/
IL void Dfs(RG int u, RG int ff){
eul[0][++cnt] = deep[u]; id[u] = cnt;
for(RG int e = fst[u]; e != -1; e = nxt[e]){
if(to[e] == ff) continue;
deep[to[e]] = deep[u] + 1;
Dfs(to[e], u);
eul[0][++cnt] = deep[u];
}
} IL int Query(RG int x, RG int y){
x = id[x]; y = id[y]; if(x > y) swap(x, y); RG int len = y - x + 1;
return min(eul[lg[len]][x], eul[lg[len]][y - pw[lg[len]] + 1]);
} IL int MaxDis(RG int x, RG int y){ return deep[x] + deep[y] - 2 * Query(x, y); } IL void Close(RG int x, RG int y){
if(x == y){
B[x].Push(0);
if(B[x].Size() == 2) A.Push(B[x].Top());
}
if(!frt[x]) return;
RG int ff = frt[x], dis = MaxDis(ff, y), tmp = C[x].Top(); C[x].Push(dis);
if(dis > tmp){
RG int mmx = B[ff].Top() + B[ff]._Top(), sszz = B[ff].Size();
if(tmp) B[ff].Del(tmp); B[ff].Push(dis);
RG int _mmx = B[ff].Top() + B[ff]._Top();
if(_mmx > mmx){
if(sszz >= 2) A.Del(mmx);
if(B[ff].Size() >= 2) A.Push(_mmx);
}
}
Close(ff, y);
} IL void Open(RG int x, RG int y){
if(x == y){
if(B[x].Size() == 2) A.Del(B[x].Top());
B[x].Del(0);
}
if(!frt[x]) return;
RG int ff = frt[x], dis = MaxDis(ff, y), tmp = C[x].Top(); C[x].Del(dis);
if(dis == tmp){
RG int mmx = B[ff].Top() + B[ff]._Top(), sszz = B[ff].Size();
B[ff].Del(dis); if(C[x].Top()) B[ff].Push(C[x].Top());
RG int _mmx = B[ff].Top() + B[ff]._Top();
if(_mmx < mmx){
if(sszz >= 2) A.Del(mmx);
if(B[ff].Size() >= 2) A.Push(_mmx);
}
}
Open(ff, y);
} int main(RG int argc, RG char* argv[]){
Fill(fst, -1); tot = sz = n = Read(); mx[0] = 2e9;
for(RG int i = 2; i < _; ++i) lg[i] = lg[i >> 1] + 1;
for(RG int i = 1; i < 20; ++i) pw[i] = pw[i - 1] << 1;
for(RG int i = 1, u, v; i < n; i++) u = Read(), v = Read(), Add(u, v), Add(v, u);
cnt = 0; Dfs(1, 0);
for(RG int i = 1; i <= lg[cnt]; ++i)
for(RG int j = 1; j + pw[i] - 1 <= cnt; ++j)
eul[i][j] = min(eul[i - 1][j], eul[i - 1][j + pw[i - 1]]);
Getroot(1, 0); Create(rt, 0);
for(RG int i = 1; i <= n; ++i) C[i].Push(0), cls[i] = 1, Close(i, i);
for(RG int Q = Read(); Q; --Q){
RG char op; RG int x; scanf(" %c", &op);
if(op == 'G'){
if(tot <= 1) printf("%d\n", tot - 1);
else printf("%d\n", A.Top());
}
else{
x = Read();
if(cls[x]) Open(x, x), --tot, cls[x] = 0;
else Close(x, x), ++tot, cls[x] = 1;
}
}
return 0;
}

动态点分治:Bzoj1095: [ZJOI2007]Hide 捉迷藏的更多相关文章

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

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

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

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

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

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

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

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

  5. 2019.01.10 bzoj1095: [ZJOI2007]Hide 捉迷藏(动态点分治)

    传送门 蒟蒻真正意义上做的第一道动态点分治! 题意:给一棵最开始所有点都是黑点的树,支持把点的颜色变成从黑/白色变成白/黑色,问当前状态树上两个最远黑点的距离. 思路: 首先考虑不带修改一次点分治怎么 ...

  6. bzoj1095: [ZJOI2007]Hide 捉迷藏 动态点分治学习

    好迷啊...感觉动态点分治就是个玄学,蜜汁把树的深度缩到logn (静态)点分治大概是递归的时候分类讨论: 1.答案经过当前点,暴力(雾)算 2.答案不经过当前点,继续递归 由于原树可以长的奇形怪状( ...

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

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

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

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

  9. BZOJ1095 [ZJOI2007]Hide 捉迷藏

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

随机推荐

  1. 低版本IE内核浏览器兼容placeholder属性解决办法

    最简便的一个方法,通过js实现. <input type="text" name="username" id="username" v ...

  2. 借助Maven入手Spring Boot第一个程序

    目前网上有不少Spring Boot的入门文章,都很有帮助,本人最近在深入学习Spring Cloud,在搭建第一个Hello World程序时,感觉对于新手而言,介绍文章怎么详细都不为过,因为其中坑 ...

  3. vue框架-学习记录

    前段时间在做vue项目时,遇到挺多问题,想简单总结一下: 1.关于父组件,子组件的通信 网上有很多这方面的讲解,讲解也比较细致,我主要总结了自己在项目中需要的: [1]父组件-子组件 也就是" ...

  4. 并行执行 Job - 每天5分钟玩转 Docker 容器技术(134)

    有时,我们希望能同时运行多个 Pod,提高 Job 的执行效率.这个可以通过 parallelism 设置. 这里我们将并行的 Pod 数量设置为 2,实践一下: Job 一共启动了两个 Pod,而且 ...

  5. C++ 函数模板“偏特化”

         模板是C++中很重要的一个特性,利用模板可以编写出类型无关的通用代码,极大的减少了代码量,提升工作效率.C++中包含类模板.函数模板,对于需要特殊处理的类型,可以通过特化的方式来实现特定类型 ...

  6. Android Native App自动化测试实战讲解(上)(基于python)

    1.Native App自动化测试及Appuim框架介绍 android平台提供了一个基于java语言的测试框架uiautomator,它一个测试的Java库,包含了创建UI测试的各种API和执行自动 ...

  7. python爬取快手视频 多线程下载

    就是为了兴趣才搞的这个,ok 废话不多说 直接开始. 环境: python 2.7 + win10 工具:fiddler postman 安卓模拟器 首先,打开fiddler,fiddler作为htt ...

  8. Linux常用命令(精选)

    chmod -R 777 文件夹名       // -R表示递归给文件及文件夹内文件更改权限,r(4),w(2),x(1),chmod -a+rwx / chmod -u+w -g+r -o +x ...

  9. nyoj358 取石子(五) 斐波那契博弈

    我写代码找的规律:如果这个n是斐波那契数,那么它是P态,如2,3,5,8..... 找规律的代码: #include <cstdio> #include <cmath> #in ...

  10. mysql主键,外键,索引

    主键 唯一而非空,只能有一个 作用: 1.唯一的标识一行  2.作为一个可以被外键有效引用的对象  3.保证数据完整性 设计原则: 1. 主键应当是对用户没有意义的.如果用户看到了一个表示多对多关系的 ...