[SDOI2018]战略游戏 圆方树,树链剖分
Posted cj-chd
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[SDOI2018]战略游戏 圆方树,树链剖分相关的知识,希望对你有一定的参考价值。
[SDOI2018]战略游戏
这题是道路相遇(题解)的升级版,询问的两个点变成了(S)个点。
还是先建出圆方树,考虑对于询问的(S)个点,答案就是圆方树上能包含这些点的最小连通块中的圆点个数减去(S)。问题变成了怎样求这样的连通块中的圆点个数,直接给结论吧:先搞出树的dfs序,把询问的点按dfs序从小到大排一遍序,每次把答案加上第(i)和第(i + 1)个点之间的圆点个数,但是不算lca,再加上第(1)个和第(S)个点之间的圆点个数,然后除以二就得到了这个连通块内不包括整个连通块的lca的圆点个数,可以证明这个连通块内除了lca的所有点都被算了两次,最后判断一下lca是不是圆点,减去(S)就是答案。
实测树剖比倍增快很多。
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define R register
#define I inline
#define B 10000000
using namespace std;
const int N = 400003;
char buf[B], *p1, *p2;
I char gc() { return p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, B, stdin), p1 == p2) ? EOF : *p1++; }
I int rd() {
R int f = 0; R char c = gc();
while (c < 48 || c > 57)
c = gc();
while (c > 47 && c < 58)
f = f * 10 + (c ^ 48), c = gc();
return f;
}
int h[N], H[N], sta[N], dfn[N], low[N], vis[N], fa[N], dep[N], siz[N], son[N], top[N], dis[N], q[N], n, c, tim, cnt, stp;
struct edge { int s, g; }e[N], E[N];
I void add(int x, int y) { e[++c] = (edge){h[x], y}, h[x] = c; }
I void Add(int x, int y) { E[++c] = (edge){H[x], y}, H[x] = c; }
I int min(int x, int y) { return x < y ? x : y; }
I int cmp(int x, int y) { return dfn[x] < dfn[y]; }
void dfs(int x) {
vis[sta[++stp] = x] = 1, dfn[x] = low[x] = ++tim;
for (R int i = h[x], y, z; i; i = e[i].s)
if (!dfn[y = e[i].g]) {
dfs(y), low[x] = min(low[x], low[y]);
if (low[y] >= dfn[x]) {
Add(++cnt, x), Add(x, cnt);
do {
vis[z = sta[stp--]] = 0, Add(cnt, z), Add(z, cnt);
} while (z ^ y);
}
}
else
low[x] = min(low[x], dfn[y]);
}
void dfs1(int x, int f) {
fa[x] = f, dep[x] = dep[f] + 1, siz[x] = 1, dis[x] = dis[f] + (x <= n);
for (R int i = H[x], y, m = 0; i; i = E[i].s)
if ((y = E[i].g) ^ f) {
dfs1(y, x), siz[x] += siz[y];
if (siz[y] > m)
m = siz[x], son[x] = y;
}
}
void dfs2(int x, int r) {
dfn[x] = ++tim, top[x] = r;
if (son[x])
dfs2(son[x], r);
for (R int i = H[x], y; i; i = E[i].s)
if ((y = E[i].g) ^ fa[x] && y ^ son[x])
dfs2(y, y);
}
I int lca(int x, int y) {
while (top[x] ^ top[y])
dep[top[x]] > dep[top[y]] ? x = fa[top[x]] : y = fa[top[y]];
return dep[x] < dep[y] ? x : y;
}
I int query(int x, int y) { return dis[x] + dis[y] - (dis[lca(x, y)] << 1); }
int main() {
R int T = rd(), m, Q, S, i, x, y, ans;
while (T--) {
memset(h, 0, sizeof h), memset(H, 0, sizeof H), memset(son, 0, sizeof son), memset(dfn, 0, sizeof dfn);
cnt = n = rd(), m = rd(), c = 0;
for (i = 1; i <= m; ++i)
x = rd(), y = rd(), add(x, y), add(y, x);
c = tim = stp = 0, dfs(1), tim = 0, dfs1(1, 0), dfs2(1, 1), Q = rd();
while (Q--) {
S = rd();
for (i = 1; i <= S; ++i)
q[i] = rd();
sort(q + 1, q + S + 1, cmp), ans = query(q[1], q[S]);
for (i = 2; i <= S; ++i)
ans += query(q[i - 1], q[i]);
printf("%d
", (ans >> 1) - S + (lca(q[1], q[S]) <= n));
}
}
return 0;
}
以上是关于[SDOI2018]战略游戏 圆方树,树链剖分的主要内容,如果未能解决你的问题,请参考以下文章