动态树分治
动态点分治
动态点分治用来解决 带点权/边权修改 的树上路径信息统计问题。
点分树
回顾点分治的计算过程。
对于一个结点
对于一个子树中简单路径的计算,我们选择一个分治中心
可以证明,当我们每次选择连通块的重心作为分治中心的时候,点分树的深度最小,为
由于树的形态在动态点分治的过程中不会改变,所以点分树的形态在动态点分治的过程中也不会改变。
下面给出求点分树的参考代码:
void calcsiz(int x, int f) {
siz[x] = 1;
maxx[x] = 0;
for (int j = h[x]; j; j = nxt[j])
if (p[j] != f && !vis[p[j]]) {
calcsiz(p[j], x);
siz[x] += siz[p[j]];
maxx[x] = max(maxx[x], siz[p[j]]);
}
maxx[x] =
max(maxx[x], sum - siz[x]); // maxx[x] 表示以 x 为根时的最大子树大小
if (maxx[x] < maxx[rt])
rt = x; // 这里不能写 <= ,保证在第二次 calcsiz 时 rt 不改变
}
void pre(int x) {
vis[x] = true; // 表示在之后的过程中不考虑 x 这个点
for (int j = h[x]; j; j = nxt[j])
if (!vis[p[j]]) {
sum = siz[p[j]];
rt = 0;
maxx[rt] = inf;
calcsiz(p[j], -1);
calcsiz(rt, -1); // 计算两次,第二次求出以 rt 为根时的各子树大小
fa[rt] = x;
pre(rt); // 记录点分树上的父亲
}
}
int main() {
sum = n;
rt = 0;
maxx[rt] = inf;
calcsiz(1, -1);
calcsiz(rt, -1);
pre(rt);
}
实现修改
在查询和修改的时候,我们在点分树上暴力跳父亲修改。由于点分树的深度最多是
在动态点分治的过程中,需要一个结点到其点分树上的祖先的距离等其他信息,由于一个点最多有
在动态点分治的过程中,一个结点在其点分树上的祖先结点的信息中可能会被重复计算,这是我们需要消去重复部分的影响。一般的方法是对于一个连通块用两种方式记录:一个是其到分治中心的距离信息,另一个是其到点分树上分治中心父亲的距离信息。这一部分内容将在例题中得到展现。
求出点分树,对于每个结点
我们可以根据上面的定义维护
现在我们来看一下,当我们反转一个点的颜色时,
假如我们要反转结点
参考代码:
#include <algorithm>
#include <cstring>
#include <iostream>
#include <queue>
using namespace std;
constexpr int MAXN = 100010;
constexpr int inf = 2e9;
int n, a, b, m, x, col[MAXN];
// 0 off 1 on
char op;
int cur, h[MAXN * 2], nxt[MAXN * 2], p[MAXN * 2];
void add_edge(int x, int y) {
cur++;
nxt[cur] = h[x];
h[x] = cur;
p[cur] = y;
}
bool vis[MAXN];
int rt, sum, siz[MAXN], maxx[MAXN], fa[MAXN], dep[MAXN];
void calcsiz(int x, int f) {
siz[x] = 1;
maxx[x] = 0;
for (int j = h[x]; j; j = nxt[j])
if (p[j] != f && !vis[p[j]]) {
calcsiz(p[j], x);
siz[x] += siz[p[j]];
maxx[x] = max(maxx[x], siz[p[j]]);
}
maxx[x] =
max(maxx[x], sum - siz[x]); // maxx[x] 表示以 x 为根时的最大子树大小
if (maxx[x] < maxx[rt])
rt = x; // 这里不能写 <= ,保证在第二次 calcsiz 时 rt 不改变
}
struct heap {
priority_queue<int> A, B; // heap=A-B
void insert(int x) { A.push(x); }
void erase(int x) { B.push(x); }
int top() {
while (!B.empty() && A.top() == B.top()) A.pop(), B.pop();
return A.top();
}
void pop() {
while (!B.empty() && A.top() == B.top()) A.pop(), B.pop();
A.pop();
}
int top2() {
int t = top(), ret;
pop();
ret = top();
A.push(t);
return ret;
}
int size() { return A.size() - B.size(); }
} dist[MAXN], ch[MAXN], ans;
void dfs(int x, int f, int d, heap& y) {
y.insert(d);
for (int j = h[x]; j; j = nxt[j])
if (p[j] != f && !vis[p[j]]) dfs(p[j], x, d + 1, y);
}
void pre(int x) {
vis[x] = true; // 不考虑 x
for (int j = h[x]; j; j = nxt[j])
if (!vis[p[j]]) {
rt = 0;
maxx[rt] = inf;
sum = siz[p[j]];
calcsiz(p[j], -1);
calcsiz(rt, -1); // 计算两次,第二次求出以 rt 为根时的各子树大小
fa[rt] = x;
dfs(p[j], -1, 1, dist[rt]);
ch[x].insert(dist[rt].top());
dep[rt] = dep[x] + 1;
pre(rt); // 记录点分树上的父亲
}
ch[x].insert(0);
if (ch[x].size() >= 2)
ans.insert(ch[x].top() + ch[x].top2());
else if (ch[x].size())
ans.insert(ch[x].top());
}
struct LCA {
int dep[MAXN], lg[MAXN], fa[MAXN][20];
void dfs(int x, int f) {
for (int j = h[x]; j; j = nxt[j])
if (p[j] != f) dep[p[j]] = dep[x] + 1, fa[p[j]][0] = x, dfs(p[j], x);
}
void init() {
dfs(1, -1);
for (int i = 2; i <= n; i++) lg[i] = lg[i / 2] + 1;
for (int j = 1; j <= lg[n]; j++)
for (int i = 1; i <= n; i++) fa[i][j] = fa[fa[i][j - 1]][j - 1];
}
int query(int x, int y) {
if (dep[x] > dep[y]) swap(x, y);
int k = dep[y] - dep[x];
for (int i = 0; k; k = k / 2, i++)
if (k & 1) y = fa[y][i];
if (x == y) return x;
k = dep[x];
for (int i = lg[k]; i >= 0; i--)
if (fa[x][i] != fa[y][i]) x = fa[x][i], y = fa[y][i];
return fa[x][0];
}
int dist(int x, int y) { return dep[x] + dep[y] - 2 * dep[query(x, y)]; }
} lca;
int d[MAXN][20];
int main() {
cin.tie(nullptr)->sync_with_stdio(false);
cin >> n;
for (int i = 1; i < n; i++) cin >> a >> b, add_edge(a, b), add_edge(b, a);
lca.init();
rt = 0;
maxx[rt] = inf;
sum = n;
calcsiz(1, -1);
calcsiz(rt, -1);
pre(rt);
for (int i = 1; i <= n; i++)
for (int j = i; j; j = fa[j]) d[i][dep[i] - dep[j]] = lca.dist(i, j);
cin >> m;
while (m--) {
cin >> op;
if (op == 'G') {
if (ans.size())
cout << ans.top() << '\n';
else
cout << "-1\n";
} else {
cin >> x;
if (!col[x]) {
if (ch[x].size() >= 2) ans.erase(ch[x].top() + ch[x].top2());
ch[x].erase(0);
if (ch[x].size() >= 2) ans.insert(ch[x].top() + ch[x].top2());
for (int i = x; fa[i]; i = fa[i]) {
if (ch[fa[i]].size() >= 2)
ans.erase(ch[fa[i]].top() + ch[fa[i]].top2());
ch[fa[i]].erase(dist[i].top());
dist[i].erase(d[x][dep[x] - dep[fa[i]]]);
if (dist[i].size()) ch[fa[i]].insert(dist[i].top());
if (ch[fa[i]].size() >= 2)
ans.insert(ch[fa[i]].top() + ch[fa[i]].top2());
}
} else {
if (ch[x].size() >= 2) ans.erase(ch[x].top() + ch[x].top2());
ch[x].insert(0);
if (ch[x].size() >= 2) ans.insert(ch[x].top() + ch[x].top2());
for (int i = x; fa[i]; i = fa[i]) {
if (ch[fa[i]].size() >= 2)
ans.erase(ch[fa[i]].top() + ch[fa[i]].top2());
if (dist[i].size()) ch[fa[i]].erase(dist[i].top());
dist[i].insert(d[x][dep[x] - dep[fa[i]]]);
ch[fa[i]].insert(dist[i].top());
if (ch[fa[i]].size() >= 2)
ans.insert(ch[fa[i]].top() + ch[fa[i]].top2());
}
}
col[x] ^= 1;
}
}
return 0;
}
我们用动态开点权值线段树记录距离信息。
类似于上题的思路,对于每个结点,我们维护线段树
在本题中,所有查询和修改都需要在点分树上对所有祖先进行修改。
以查询操作为例,如果我们要查询距离结点
在进行修改操作时,我们要同时维护
参考代码:
#include <algorithm>
#include <cstring>
#include <iostream>
using namespace std;
constexpr int MAXN = 100010;
constexpr int inf = 2e9;
constexpr int ddd = 6000010;
struct Segtree {
int cnt, rt[MAXN], sum[ddd], lc[ddd], rc[ddd];
void update(int& o, int l, int r, int x, int v) {
if (!o) o = ++cnt;
if (l == r) {
sum[o] += v;
return;
}
int mid = (l + r) >> 1;
if (x <= mid)
update(lc[o], l, mid, x, v);
else
update(rc[o], mid + 1, r, x, v);
sum[o] = sum[lc[o]] + sum[rc[o]];
}
int query(int o, int l, int r, int ql, int qr) {
if (!o || r < ql || l > qr) return 0;
if (ql <= l && r <= qr) return sum[o];
int mid = (l + r) >> 1;
return query(lc[o], l, mid, ql, qr) + query(rc[o], mid + 1, r, ql, qr);
}
} dist, ch;
int n, m, val[MAXN], u, v, op, x, y, lstans;
int cur, h[MAXN * 2], nxt[MAXN * 2], p[MAXN * 2];
void add_edge(int x, int y) {
cur++;
nxt[cur] = h[x];
h[x] = cur;
p[cur] = y;
}
struct LCA {
int dep[MAXN], lg[MAXN], fa[MAXN][20];
void dfs(int x, int f) {
for (int j = h[x]; j; j = nxt[j])
if (p[j] != f) dep[p[j]] = dep[x] + 1, fa[p[j]][0] = x, dfs(p[j], x);
}
void init() {
dep[1] = 1;
dfs(1, -1);
for (int i = 2; i <= n; i++) lg[i] = lg[i / 2] + 1;
for (int j = 1; j <= lg[n]; j++)
for (int i = 1; i <= n; i++) fa[i][j] = fa[fa[i][j - 1]][j - 1];
}
int query(int x, int y) {
if (dep[x] > dep[y]) swap(x, y);
int k = dep[y] - dep[x];
for (int i = 0; k; k = k / 2, i++)
if (k & 1) y = fa[y][i];
if (x == y) return x;
k = dep[x];
for (int i = lg[k]; i >= 0; i--)
if (fa[x][i] != fa[y][i]) x = fa[x][i], y = fa[y][i];
return fa[x][0];
}
int dist(int x, int y) { return dep[x] + dep[y] - 2 * dep[query(x, y)]; }
} lca;
int rt, sum, siz[MAXN], maxx[MAXN], fa[MAXN];
int d[MAXN][20], dep[MAXN];
bool vis[MAXN];
void calcsiz(int x, int fa) {
siz[x] = 1;
maxx[x] = 0;
for (int j = h[x]; j; j = nxt[j])
if (p[j] != fa && !vis[p[j]]) {
calcsiz(p[j], x);
siz[x] += siz[p[j]];
maxx[x] = max(maxx[x], siz[p[j]]);
}
maxx[x] =
max(maxx[x], sum - siz[x]); // maxx[x] 表示以 x 为根时的最大子树大小
if (maxx[x] < maxx[rt])
rt = x; // 这里不能写 <= ,保证在第二次 calcsiz 时 rt 不改变
}
void dfs1(int x, int fa, int y, int d) {
ch.update(ch.rt[y], 0, n, d, val[x]);
for (int j = h[x]; j; j = nxt[j])
if (p[j] != fa && !vis[p[j]]) dfs1(p[j], x, y, d + 1);
}
void dfs2(int x, int fa, int y, int d) {
dist.update(dist.rt[y], 0, n, d, val[x]);
for (int j = h[x]; j; j = nxt[j])
if (p[j] != fa && !vis[p[j]]) dfs2(p[j], x, y, d + 1);
}
void pre(int x) {
vis[x] = true; // 表示在之后的过程中不考虑 x 这个点
dfs2(x, -1, x, 0);
for (int j = h[x]; j; j = nxt[j])
if (!vis[p[j]]) {
rt = 0;
maxx[rt] = inf;
sum = siz[p[j]];
calcsiz(p[j], -1);
calcsiz(rt, -1); // 计算两次,第二次求出以 rt 为根时的各子树大小
dfs1(p[j], -1, rt, 1);
fa[rt] = x;
dep[rt] = dep[x] + 1;
pre(rt); // 记录点分树上的父亲
}
}
int main() {
cin.tie(nullptr)->sync_with_stdio(false);
cin >> n >> m;
for (int i = 1; i <= n; i++) cin >> val[i];
for (int i = 1; i < n; i++) cin >> u >> v, add_edge(u, v), add_edge(v, u);
lca.init();
rt = 0;
maxx[rt] = inf;
sum = n;
calcsiz(1, -1);
calcsiz(rt, -1);
pre(rt);
for (int i = 1; i <= n; i++)
for (int j = i; j; j = fa[j]) d[i][dep[i] - dep[j]] = lca.dist(i, j);
while (m--) {
cin >> op >> x >> y;
x ^= lstans;
y ^= lstans;
if (op == 0) {
lstans = dist.query(dist.rt[x], 0, n, 0, y);
int nww = 0;
for (int i = x; fa[i]; i = fa[i]) {
nww = d[x][dep[x] - dep[fa[i]]]; // lca.dist(x,fa[i]);
lstans += dist.query(dist.rt[fa[i]], 0, n, 0, y - nww);
lstans -= ch.query(ch.rt[i], 0, n, 0, y - nww);
}
cout << lstans << '\n';
}
if (op == 1) {
int nww = 0;
dist.update(dist.rt[x], 0, n, 0, y - val[x]);
for (int i = x; fa[i]; i = fa[i]) {
nww = d[x][dep[x] - dep[fa[i]]]; // lca.dist(x,fa[i]);
dist.update(dist.rt[fa[i]], 0, n, nww, y - val[x]);
ch.update(ch.rt[i], 0, n, nww, y - val[x]);
}
val[x] = y;
}
}
return 0;
}
创建日期: 2018年7月11日