题面

洛谷

题解

代码

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
inline int gi() {
register int data = 0, w = 1;
register char ch = 0;
while (!isdigit(ch) && ch != '-') ch = getchar();
if (ch == '-') w = -1, ch = getchar();
while (isdigit(ch)) data = 10 * data + ch - '0', ch = getchar();
return w * data;
}
const int MAX_N = 3e5 + 5;
struct Graph { int to, next; } e[MAX_N << 1]; int fir[MAX_N], e_cnt;
void clearGraph() { memset(fir, -1, sizeof(fir)); e_cnt = 0; }
void Add_Edge(int u, int v) { e[e_cnt] = (Graph){v, fir[u]}, fir[u] = e_cnt++; }
int bin[20];
int N, M, dfn;
int fa[MAX_N][20], dep[MAX_N], id[MAX_N], size[MAX_N];
int a[MAX_N], b[MAX_N], f[MAX_N];
int top, stk[MAX_N], c[MAX_N];
int rem[MAX_N], bel[MAX_N];
bool cmp(int a, int b) { return id[a] < id[b]; }
void dfs(int x, int f) {
id[x] = ++dfn;
size[x] = 1;
for (int i = 1; bin[i] <= dep[x]; i++) fa[x][i] = fa[fa[x][i - 1]][i - 1];
for (int i = fir[x]; ~i; i = e[i].next)
if (e[i].to != f) {
fa[e[i].to][0] = x;
dep[e[i].to] = dep[x] + 1;
dfs(e[i].to, x);
size[x] += size[e[i].to];
}
}
int LCA(int x, int y) {
if (dep[x] > dep[y]) swap(x, y);
int t = dep[y] - dep[x];
for (int i = 0; bin[i] <= t; i++)
if (bin[i] & t) y = fa[y][i];
for (int i = 18; i >= 0; i--)
if (fa[x][i] != fa[y][i]) x = fa[x][i], y = fa[y][i];
return x == y ? x : fa[x][0];
}
int dis(int a, int b) { return dep[a] + dep[b] - 2 * dep[LCA(a, b)]; }
void solve(int a, int b) {
int x = b, mid = b;
for (int i = 18; i >= 0; i--)
if (dep[fa[x][i]] > dep[a]) x = fa[x][i];
rem[a] -= size[x];
if (bel[a] == bel[b]) {
f[bel[a]] += size[x] - size[b];
return ;
}
for (int i = 18; i >= 0; i--) {
int nxt = fa[mid][i]; if (dep[nxt] <= dep[a]) continue;
int t1 = dis(bel[a], nxt), t2 = dis(bel[b], nxt);
if (t1 > t2 || (t1 == t2 && bel[b] < bel[a])) mid = nxt;
}
f[bel[a]] += size[x] - size[mid];
f[bel[b]] += size[mid] - size[b];
}
void dfs1(int x) {
c[++dfn] = x, rem[x] = size[x];
for (int i = fir[x]; ~i; i = e[i].next) {
dfs1(e[i].to); if (!bel[e[i].to]) continue;
int t1 = dis(bel[e[i].to], x), t2 = dis(bel[x], x);
if ((t1 == t2 && bel[e[i].to] < bel[x]) || t1 < t2 || !bel[x])
bel[x] = bel[e[i].to];
}
}
void dfs2(int x) {
for (int i = fir[x]; ~i; i = e[i].next) {
int t1 = dis(bel[x], e[i].to), t2 = dis(bel[e[i].to], e[i].to);
if ((t1 == t2 && bel[e[i].to] > bel[x]) || t1 < t2 || !bel[e[i].to])
bel[e[i].to] = bel[x];
dfs2(e[i].to);
}
}
void solve() {
top = dfn = e_cnt = 0;
M = gi();
for (int i = 1; i <= M; i++) a[i] = b[i] = gi();
for (int i = 1; i <= M; i++) bel[a[i]] = a[i];
sort(&a[1], &a[M + 1], cmp);
if (bel[1] != 1) stk[++top] = 1;
for (int i = 1; i <= M; i++) {
int t = a[i], f = 0;
while (top > 0) {
f = LCA(stk[top], t);
if (top > 1 && dep[f] < dep[stk[top - 1]])
Add_Edge(stk[top - 1], stk[top]), top--;
else if (dep[f] < dep[stk[top]]) {
Add_Edge(f, stk[top]); top--;
break;
} else break;
}
if (stk[top] != f) stk[++top] = f;
stk[++top] = t;
}
while (top > 1) Add_Edge(stk[top - 1], stk[top]), top--;
dfs1(1), dfs2(1);
for (int i = 1; i <= dfn; i++)
for (int j = fir[c[i]]; ~j; j = e[j].next) solve(c[i], e[j].to);
for (int i = 1; i <= dfn; i++) f[bel[c[i]]] += rem[c[i]];
for (int i = 1; i <= M; i++) printf("%d ", f[b[i]]);
puts("");
for (int i = 1; i <= dfn; i++) f[c[i]] = bel[c[i]] = rem[c[i]] = 0, fir[c[i]] = -1;
}
int main() {
#ifndef ONLINE_JUDGE
freopen("cpp.in", "r", stdin);
#endif
clearGraph();
bin[0] = 1;
for (int i = 1; i < 20; i++) bin[i] = bin[i - 1] << 1;
N = gi();
for (int i = 1; i < N; i++) {
int u = gi(), v = gi();
Add_Edge(u, v);
Add_Edge(v, u);
}
dfs(1, 0);
clearGraph();
int Q = gi();
while (Q--) solve();
return 0;
}

【LG3233】[HNOI2014]世界树的更多相关文章

  1. [BZOJ3572][Hnoi2014]世界树

    [BZOJ3572][Hnoi2014]世界树 试题描述 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森,在他们的信条 ...

  2. BZOJ 3572: [Hnoi2014]世界树

    BZOJ 3572: [Hnoi2014]世界树 标签(空格分隔): OI-BZOJ OI-虚数 OI-树形dp OI-倍增 Time Limit: 20 Sec Memory Limit: 512 ...

  3. bzoj 3572: [Hnoi2014]世界树 虚树 && AC500

    3572: [Hnoi2014]世界树 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 520  Solved: 300[Submit][Status] ...

  4. bzoj 3572 [Hnoi2014]世界树(虚树+DP)

    3572: [Hnoi2014]世界树 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 645  Solved: 362[Submit][Status] ...

  5. 【BZOJ3572】[Hnoi2014]世界树 虚树

    [BZOJ3572][Hnoi2014]世界树 Description 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森 ...

  6. bzoj3572[Hnoi2014] 世界树 虚树+dp+倍增

    [Hnoi2014]世界树 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1921  Solved: 1019[Submit][Status][Dis ...

  7. <虚树+树型DP> HNOI2014世界树

    <虚树+树型DP> HNOI2014世界树 #include <iostream> #include <cstdio> #include <cstring&g ...

  8. [题解] [HNOI2014] 世界树

    题面 [HNOI2014]世界树 题解 从数据范围很容易看出是个虚树DP(可惜看出来了也还是不会做) 虚树大家应该都会, 不会的话自己去搜吧, 我懒得讲了, 我们在这里只需要考虑如何DP即可 首先我们 ...

  9. [HNOI2014]世界树

    题目描述 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森,在他们的信条里,公平是使世界树能够生生不息.持续运转的根本基 ...

随机推荐

  1. spring boot 输入参数统一校验

    1 引入spring boot validate    maven 依赖 <!-- 验证 --> <dependency> <groupId>org.hiberna ...

  2. 两个事务 update同一张表出现的死锁问题 (转载)

    引言 近来做省一级计算机一级考试系统的时候,学生端进行大批量判分的时候,出现了这样的问题(事务(进程 ID 262)与另一个进程被死锁在 锁 资源上,并且已被选作死锁牺牲品.请重新运行该事务.): 这 ...

  3. 解决Js跨域访问的问题

    1,最近有个需求,用Js获取Html标签<input type="file"/>的路径!遇到代码拒绝访问,提示安全验证,不允许跨域访问,简单的设置一下浏览器即可,不过对 ...

  4. SpirteKit深度复制SKSpriteNode节点及convert转换其它Scene上的节点到当前场景坐标

    Playground输出的代码 ,注意右侧打印出的SpriteNode输出值,HAPPY_NODE的输出信息一致. SpriteNode //MARK:- 扩展SkSpriteNode属性 exten ...

  5. vue中的slot(插槽)

    vue中的插槽----slot 什么是插槽? 插槽(Slot)是Vue提出来的一个概念,正如名字一样,插槽用于决定将所携带的内容,插入到指定的某个位置,从而使模板分块,具有模块化的特质和更大的重用性. ...

  6. 关于 web.config impersonate 帐号模拟

    1.模拟 IIS 验证的帐户或用户 若要在收到 ASP.NET 应用程序中每个页的每个请求时模拟 Microsoft Internet 信息服务 (IIS) 身份验证用户,必须在此应用程序的 Web. ...

  7. windows 使用npm安装webpack 4.0以及配置问题的解决办法

    输入cmd点击打开 输入node -v 出现nodejs版本号 输入npm -v 出现npm版本号则安装npm安装成功, 2.安装webpack 桌面新建一个webpack-test文件夹,点击进入文 ...

  8. 在阿里云的CentOS 6.5 上面安装 timidity++ 和 ffmpeg(含libmp3lame) 实现命令行将midi转换为mp3

    首先使用yum安装需要的组件 yum install -y automake autoconf libtool gcc gcc-c++ yasm yasm-devel 然后从sourceforge下载 ...

  9. 伪类选择器a的属性

    在使用a中的四个属性时必须按照顺序来,不管用那两个都是要依照这个顺序来 1.a:link未访问过的状态 a:link{ color:red; } 2.a:visited已访问的 a:visited{ ...

  10. 高性能MySQL--创建高性能的索引

    关于MySQL的优化,相信很多人都听过这一条:避免使用select *来查找字段,而是要在select后面写上具体的字段. 那么这么做的原因相信大家都应该知道:减少数据量的传输. 但我要讲的是另外一个 ...