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

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm> using namespace std; typedef long long LL;
const int MAXN = 3e5 + 10; inline int in()
{
int x = 0, flag = 1; char ch = getchar();
while (ch < '0' || ch > '9') { if (ch == '-') flag = -1; ch = getchar(); }
while (ch >= '0' && ch <= '9') x = (x << 1) + (x << 3) + (ch ^ 48), ch = getchar();
return x * flag;
} int n, m, q; struct Gra
{
int head[MAXN], nume;
struct Adj { int nex, to; } adj[MAXN << 1];
void clear()
{
memset(head, 0, sizeof head);
nume = 0;
}
void addedge(int from, int to)
{
adj[++nume] = (Adj) { head[from], to };
head[from] = nume;
}
void link(int from, int to)
{
addedge(from, to);
addedge(to, from);
}
} g1, g2; int lg[MAXN], up[20][MAXN], dfn[MAXN], ind, dep[MAXN], size[MAXN];
void DFS(int u)
{
dfn[u] = ++ind;
dep[u] = dep[up[0][u]] + 1;
size[u] = 1;
for (int i = 1; (1 << i) <= dep[u]; ++ i)
up[i][u] = up[i - 1][up[i - 1][u]];
for (int i = g1.head[u]; i; i = g1.adj[i].nex)
{
int v = g1.adj[i].to;
if (v == up[0][u]) continue;
up[0][v] = u;
DFS(v);
size[u] += size[v];
}
}
int lca(int x, int y)
{
if (dep[x] > dep[y]) swap(x, y);
while (dep[x] != dep[y]) y = up[lg[dep[y] - dep[x]]][y];
if (x == y) return x;
for (int i = lg[dep[x]]; i >= 0; -- i)
if (up[i][x] != up[i][y]) x = up[i][x], y = up[i][y];
return up[0][x];
} int a[MAXN], id[MAXN], ans[MAXN];
bool cmpd(int x, int y) { return dfn[x] < dfn[y]; } int stk[MAXN], top;
bool iskey[MAXN]; void insert(int x)
{
if (x == stk[1]) return;
int LCA = lca(x, stk[top]);
if (LCA == stk[top]) return (void)(stk[++top] = x);
while (top > 1 && dep[LCA] <= dep[stk[top - 1]])
{
g2.addedge(stk[top - 1], stk[top]);
-- top;
}
if (LCA != stk[top]) g2.addedge(LCA, stk[top]), stk[top] = LCA;
stk[++top] = x;
} struct Mn
{
int id, v;
} mn[MAXN];
void getmin(int x, int y)
{
int val = mn[y].v + abs(dep[x] - dep[y]);
if (val < mn[x].v)
mn[x] = (Mn) { mn[y].id, val };
else if (val == mn[x].v && mn[y].id < mn[x].id)
mn[x].id = mn[y].id;
}
void DFS1(int u)
{
if (iskey[u]) mn[u] = (Mn) { u, 0 };
else mn[u] = (Mn) { 300001, 2333333 };
for (int i = g2.head[u]; i; i = g2.adj[i].nex)
{
int v = g2.adj[i].to;
DFS1(v);
getmin(u, v);
}
return ;
}
int jump(int x, int d)
{
for (int i = 19; i >= 0; -- i)
if ((1 << i) <= d) x = up[i][x], d -= (1 << i);
return x;
}
void DFS2(int u)
{
ans[mn[u].id] += 1;
int sum = 0;
for (int i = g2.head[u]; i; i = g2.adj[i].nex)
{
int v = g2.adj[i].to;
getmin(v, u);
int dis = dep[v] - dep[u], mid = (mn[u].v + mn[v].v + dis) / 2;
int uson = jump(v, dis - 1);
sum += size[uson];
if (mid - mn[v].v > 0 && mid - mn[v].v < dis)
{
int mi = jump(v, mid - mn[v].v), mison = jump(v, mid - mn[v].v - 1);
ans[mn[v].id] += size[mison] - size[v];
ans[mn[u].id] += size[uson] - size[mi];
if (mid * 2 != mn[u].v + mn[v].v + dis) ans[mn[v].id] += size[mi] - size[mison];
else
{
if (mn[u].id < mn[v].id) ans[mn[u].id] += size[mi] - size[mison];
else ans[mn[v].id] += size[mi] - size[mison];
}
}
else if (mid - mn[v].v <= 0) ans[mn[u].id] += size[uson] - size[v];
else if (mid - mn[v].v >= dis) ans[mn[v].id] += size[uson] - size[v];
DFS2(v);
}
ans[mn[u].id] += size[u] - sum - 1;
g2.head[u] = 0;
iskey[u] = false;
return ;
} int main()
{
g1.clear(); g2.clear();
n = in();
for (int i = 1; i <= n; ++ i)
lg[i] = lg[i - 1] + ((2 << lg[i - 1]) == i);
for (int i = 1; i < n; ++ i)
{
int u = in(), v = in();
g1.link(u, v);
}
DFS(1);
q = in();
while (q --)
{
m = in();
g2.nume = 0;
for (int i = 1; i <= m; ++ i) a[i] = in(), id[i] = a[i];
sort(a + 1, a + m + 1, cmpd);
top = 0; stk[++ top] = 1;
for (int i = 1; i <= m; ++ i) insert(a[i]), iskey[a[i]] = true;
while (top > 1) g2.addedge(stk[top - 1], stk[top]), -- top;
DFS1(1);
DFS2(1);
for (int i = 1; i <= m; ++ i) printf("%d ", ans[id[i]]); puts("");
for (int i = 1; i <= m; ++ i) ans[id[i]] = 0;
}
return 0;
}

<虚树+树型DP> HNOI2014世界树的更多相关文章

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

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

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

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

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

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

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

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

  5. BZOJ 2286 消耗战 - 虚树 + 树型dp

    传送门 题目大意: 每次给出k个特殊点,回答将这些特殊点与根节点断开至少需要多少代价. 题目分析: 虚树入门 + 树型dp: 刚刚学习完虚树(好文),就来这道入门题签个到. 虚树就是将树中的一些关键点 ...

  6. <虚树+树型DP> SDOI2011消耗战

    <虚树+树型DP> SDOI2011消耗战 #include <iostream> #include <cstdio> #include <cstring&g ...

  7. POJ3659 Cell Phone Network(树上最小支配集:树型DP)

    题目求一棵树的最小支配数. 支配集,即把图的点分成两个集合,所有非支配集内的点都和支配集内的某一点相邻. 听说即使是二分图,最小支配集的求解也是还没多项式算法的.而树上求最小支配集树型DP就OK了. ...

  8. POJ 3342 - Party at Hali-Bula 树型DP+最优解唯一性判断

    好久没写树型dp了...以前都是先找到叶子节点.用队列维护来做的...这次学着vector动态数组+DFS回朔的方法..感觉思路更加的清晰... 关于题目的第一问...能邀请到的最多人数..so ea ...

  9. bzoj 2286: [Sdoi2011]消耗战 虚树+树dp

    2286: [Sdoi2011]消耗战 Time Limit: 20 Sec  Memory Limit: 512 MB[Submit][Status][Discuss] Description 在一 ...

随机推荐

  1. jquery取消绑定的方法

    jquery取消绑定的方法 一般用变量控制 不要用unbind() 相应比较慢 <pre> $('.choseitem').on('click', function () { //如果设置 ...

  2. PHP接口并发测试的方法

    PHP接口并发测试的方法 <pre> header('Content-type:text/html; Charset=utf-8'); $uri = "输入你的url" ...

  3. Linux内核中的并发与竞态概述

    1.前言 众所周知,Linux系统是一个多任务的操作系统,当多个任务同时访问同一片内存区域的时候,这些任务可能会相互覆盖内存中数据,从而造成内存中的数据混乱,问题严重的话,还可能会导致系统崩溃. 2. ...

  4. Redis 内存管理 源码分析

    要想了解redis底层的内存管理是如何进行的,直接看源码绝对是一个很好的选择 下面是我添加了详细注释的源码,需要注意的是,为了便于源码分析,我把redis为了弥补平台差异的那部分代码删了,只需要知道有 ...

  5. skywalking集群部署

    1.需求:有两台服务器分别装了app,对这些app进行性能监控 三台服务器:10.10.20.198作为服务端展示性能数据,10.10.20.64客户端,装btam系统,10.10.20.63客户端装 ...

  6. 参加AWS技术峰会的收获与思考

    7月31日,我参加了AWS 技术峰会2019北京站的会议. 从厦门到帝都,奔赴千里,只为一场技术盛宴,我想记录一些收获和思考,才能不负此行. 大会议程全天,上午是主题演讲和行业解决方案展示,下午是技术 ...

  7. 【51nod1355】斐波那契的最小公倍数(min-max容斥)

    [51nod1355]斐波那契的最小公倍数(min-max容斥) 题面 51nod 题解 显然直接算还是没法算的,所以继续考虑\(min-max\)容斥计算. \[lcm(S)=\prod_{T\su ...

  8. C# LDAP认证登录类参考

    public class LDAPHelper     {         private DirectoryEntry _objDirectoryEntry;         /// <sum ...

  9. C# 递增操作符 ++ --

    记混了好几次,记录一下 递增操作符出现在操作数之前:先递增后赋值 ; int result; result =++count; Console.WriteLine($"count:{coun ...

  10. 如何简单使用tensorboard展示(二)

    我使用tensorboard继续做了标量展示与直方图展示,在一的基础做了拓展,其改写代码如下: import numpy as npimport tensorflow as tfimport rand ...