【题解】HNOI2014世界树
脑子不清醒的时候千万别写题。写题写不下去了千万别死扛,重构才是你唯一的出路QAQ
昨天很想快点写道题,思路没有很清晰的时候就写了,结果……今天一怒之下决定重整思路重构代码,其实不过是半个小时的事情……
提示很明显,总点数限制了范围。建立出虚树,在虚树上面 dp。 在虚树上面我们dp两遍,两遍一起处理出每一个(虚树上的)节点被谁管辖,在连接两者的路径上找到管理的分界点即可。处理两点之间的距离一定要上 ST 表啊……
#include <bits/stdc++.h>
using namespace std;
#define maxn 350000
#define CNST 20
int n, dep[maxn], gra[maxn][CNST];
int size[maxn], hson[maxn], ans[maxn];
int top, S[maxn], a[maxn], b[maxn];
int timer, dfn[maxn], sum[maxn];
bool mark[maxn], vis[maxn];
int T, Log[maxn * 2], bit[21], ST[maxn * 2][CNST], pos[maxn]; int read()
{
int x = 0, k = 1;
char c;
c = getchar();
while(c < '0' || c > '9') { if(c == '-') k = -1; c = getchar(); }
while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return x * k;
} struct edge
{
int cnp, to[maxn * 2], last[maxn * 2], head[maxn];
edge() { cnp = 1; }
void add(int u, int v)
{
if(u == v) return;
to[cnp] = v, last[cnp] = head[u], head[u] = cnp ++;
to[cnp] = u, last[cnp] = head[v], head[v] = cnp ++;
}
}E1, E2; bool cmp(int x, int y) { return dfn[x] < dfn[y]; } void dfs(int u, int fa)
{
size[u] = 1; dfn[u] = ++ timer;
gra[u][0] = fa; dep[u] = dep[fa] + 1; ST[++ T][0] = dep[u]; pos[u] = T;
for(int i = 1; i < CNST; i ++) gra[u][i] = gra[gra[u][i - 1]][i - 1];
for(int i = E1.head[u]; i; i = E1.last[i])
{
int v = E1.to[i];
if(v == fa) return;
dfs(v, u); size[u] += size[v];
ST[++ T][0] = dep[u];
}
} int LCA(int x, int y)
{
if(dep[x] < dep[y]) swap(x, y);
for(int i = CNST - 1; ~i; i --)
if(dep[gra[x][i]] >= dep[y]) x = gra[x][i];
for(int i = CNST - 1; ~i; i --)
if(gra[x][i] != gra[y][i]) x = gra[x][i], y = gra[y][i];
return (x == y) ? x : gra[x][0];
} int Get_Dis(int x, int y)
{
int ret = dep[x] + dep[y];
if(!x) return dep[y]; else if(!y) return dep[x];
x = pos[x], y = pos[y];
if(x > y) swap(x, y);
int k = Log[y - x + 1];
ret = ret - 2 * min(ST[x][k], ST[y - bit[k] + 1][k]);
return ret;
} int Get_Gra(int x, int y)
{
for(int i = CNST - 1; ~i; i --)
if(dep[gra[x][i]] > dep[y]) x = gra[x][i];
return x;
} int Work(int u, int v, int k)
{
bool flag = 0;
int x = u;
for(int i = CNST - 1; i >= 0; i --)
{
int tem = gra[x][i]; if(dep[tem] < dep[k]) continue;
int d1 = Get_Dis(u, tem), d2 = Get_Dis(v, tem);
if(d1 < d2) x = gra[x][i];
else if(d1 == d2 && u < v) x = gra[x][i];
}
return x;
} void DP(int u, int fa)
{
hson[u] = 0; sum[u] = 0;
for(int i = E2.head[u]; i; i = E2.last[i])
{
int v = E2.to[i]; if(!v) continue;
if(v == fa) continue;
DP(v, u);
int t = hson[v];
int dis = Get_Dis(u, t), dis2 = Get_Dis(u, hson[u]);
if(dis < dis2) hson[u] = t;
else if(dis == dis2 && t < hson[u]) hson[u] = t;
if(!hson[u]) hson[u] = t;
}
if(mark[u]) hson[u] = u;
} void DP2(int u, int fa)
{
int tot = 0;
if(hson[fa])
{
int t = hson[fa];
int dis = Get_Dis(u, t), dis2 = Get_Dis(u, hson[u]);
if(dis < dis2) hson[u] = t;
else if(dis == dis2 && t < hson[u]) hson[u] = t;
if(!hson[u]) hson[u] = t;
}
for(int i = E2.head[u]; i; i = E2.last[i])
{
int v = E2.to[i];
if(v == fa) continue; DP2(v, u);
int y = Get_Gra(v, u);
int x = Work(hson[v], hson[u], u);
int tem = size[x] - size[v];
ans[hson[v]] += tem, ans[hson[u]] += size[y] - size[x];
tot += size[y];
}
ans[hson[u]] += size[u] - tot;
E2.head[u] = 0;
} void Solve()
{
int K = read();
for(int i = 1; i <= K; i ++) b[i] = a[i] = read(), mark[a[i]] = 1;
sort(a + 1, a + 1 + K, cmp); S[top = 1] = 1; E2.cnp = 1; ans[0] = 0;
for(int i = 1; i <= K; i ++)
{
int lca = LCA(S[top], a[i]);
while(2333)
{
if(dep[S[top - 1]] <= dep[lca])
{
E2.add(lca, S[top --]);
if(lca != S[top]) S[++ top] = lca;
break;
}
else E2.add(S[top], S[top - 1]), top --;
}
S[++ top] = a[i];
}
while(top > 1) E2.add(S[top], S[top - 1]), top --;
sum[1] = 0; DP(1, 0); DP2(1, 0);
for(int i = 1; i <= K; i ++) printf("%d ", ans[b[i]]), mark[b[i]] = 0, ans[b[i]] = 0;
puts("");
} void init()
{
bit[0] = 1; for(int i = 1; i <= 20; i ++) bit[i] = bit[i - 1] << 1;
Log[0] = -1; for(int i = 1; i < maxn * 2; i ++) Log[i] = Log[i >> 1] + 1;
} int main()
{
init(); n = read();
for(int i = 1; i < n; i ++)
{
int x = read(), y = read();
E1.add(x, y);
}
dfs(1, 0);
for(int i = 1; i <= Log[T]; i ++)
for(int j = 1; j + bit[i - 1] <= T; j ++)
ST[j][i] = min(ST[j][i - 1], ST[j + bit[i - 1]][i - 1]);
int q = read();
for(int i = 1; i <= q; i ++) Solve();
return 0;
}
【题解】HNOI2014世界树的更多相关文章
- [题解] [HNOI2014] 世界树
题面 [HNOI2014]世界树 题解 从数据范围很容易看出是个虚树DP(可惜看出来了也还是不会做) 虚树大家应该都会, 不会的话自己去搜吧, 我懒得讲了, 我们在这里只需要考虑如何DP即可 首先我们 ...
- [BZOJ3572][Hnoi2014]世界树
[BZOJ3572][Hnoi2014]世界树 试题描述 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森,在他们的信条 ...
- 【BZOJ3572】[Hnoi2014]世界树 虚树
[BZOJ3572][Hnoi2014]世界树 Description 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森 ...
- bzoj3572[Hnoi2014] 世界树 虚树+dp+倍增
[Hnoi2014]世界树 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1921 Solved: 1019[Submit][Status][Dis ...
- BZOJ 3572: [Hnoi2014]世界树
BZOJ 3572: [Hnoi2014]世界树 标签(空格分隔): OI-BZOJ OI-虚数 OI-树形dp OI-倍增 Time Limit: 20 Sec Memory Limit: 512 ...
- bzoj 3572: [Hnoi2014]世界树 虚树 && AC500
3572: [Hnoi2014]世界树 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 520 Solved: 300[Submit][Status] ...
- bzoj 3572 [Hnoi2014]世界树(虚树+DP)
3572: [Hnoi2014]世界树 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 645 Solved: 362[Submit][Status] ...
- <虚树+树型DP> HNOI2014世界树
<虚树+树型DP> HNOI2014世界树 #include <iostream> #include <cstdio> #include <cstring&g ...
- BZOJ3572:[HNOI2014]世界树——题解
+++++++++++++++++++++++++++++++++++++++++++ +本文作者:luyouqi233. + +欢迎访问我的博客:http://www.cnblogs.com/luy ...
随机推荐
- LeetCode:46. Permutations(Medium)
1. 原题链接 https://leetcode.com/problems/permutations/description/ 2. 题目要求 给定一个整型数组nums,数组中的数字互不相同,返回该数 ...
- SpringBoot入门(五)——自定义配置
本文来自网易云社区 大部分比萨店也提供某种形式的自动配置.你可以点荤比萨.素比萨.香辣意大利比萨,或者是自动配置比萨中的极品--至尊比萨.在下单时,你并没有指定具体的辅料,你所点的比萨种类决定了所用的 ...
- tomcat createSecureRandom 花费了将近10分钟
http://www.th7.cn/Program/java/201603/776312.shtml 启动tomcat很慢,检查后发现:[localhost-startStop-1] org.apac ...
- 容器云技术:容器化微服务,Istio占C位出道
在精彩的软件容器世界中,当新项目涌现并解决你认为早已解决的问题时,这感觉就像地面在你的脚下不断地移动.在许多情况下,这些问题很久以前被解决,但现在的云原生架构正在推动着更大规模的应用程序部署,这就需要 ...
- 180623-SpringBoot之logback配置文件
SpringBoot配置logback 项目的日志配置属于比较常见的case了,之前接触和使用的都是Spring结合xml的方式,引入几个依赖,然后写个 logback.xml 配置文件即可,那么在S ...
- Linux命令应用大词典-第12章 程序编译
12.1 gcc:GNU项目的C和C++编译器 12.2 gdberver:为GNU调试的远程服务器 12.3 cmake:跨平台的Makefile生成工具 12.4 indent:更改通过插入或删除 ...
- Python元组与列表的区别和联系?
1. 元组和列表比较相似,不过它们之间也有着不同: (1)列表:一个大仓库,你可以随时往里边添加和删除任何东西. (2)元组:封闭的列表,一旦定义,就不可改变(不能添加.删除或修改). 2. 什么情 ...
- 前端开发工程师 - 01.页面制作 - 第3章.HTML
第3章--HTML HTML简介 Hyper Text Markup Language:超文本标记语言--用于标记网页的内容 history: html(1991)雏形 -> html4.01( ...
- ajax 和 mock 数据
ajax ajax是一种技术方案,但并不是一种新技术.它依赖的是现有的CSS/HTML/Javascript,而其中最核心的依赖是浏览器提供的XMLHttpRequest对象,是这个对象使得浏览器可以 ...
- Python基础 之 list类-列表
list类-列表 一.list类的基本属性 1. 列表格式 li = [1, 12, 9, ", 10], "even"], "root", True ...