【题意】给定n个点的树,m次询问,每次给定ki个特殊点,一个点会被最近的特殊点控制,询问每个特殊点控制多少点。n,m,Σki<=300000。

【算法】虚树+倍增

【题解】★参考:thy_asdf

对询问建立虚树,然后DFS统计虚树上每个点被哪个点控制,记为belong[x]。

统计的方法是从下往上DFS得到x来自x子树的控制点,再从上往下DFS得到x来自非x子树的控制点。

令D(x,y)表示点x和点y之间的路径长度,比较x的不同控制点y的方式是取D(x,y)的最小值(相同时比较编号)。

依此考虑虚树上的每一条边(x,y),记z为同时是x的儿子和y的祖先的点,令f[x]表示点x的答案。

如果belong[x]=belong[y],那么f[belong[x]]+=size[z]-size[y]

否则,从y倍增找到分界点mid,满足mid及以下由y控制,mid以上由x控制。

倍增过程中,记A=D(belong[x],mid),B=D(belong[x],mid),当满足A>B或A=B&&belong[x]>belong[y]时进行倍增。

找到分界点mid后,f[belong[x]]+=size[x]-size[m],f[belong[y]+=size[m]-size[y]。

最后要处理没有出现在虚树路径上的点(包括虚树节点),这些点没有在上述过程中被统计。

记rem[x]表示子树x中未被统计的节点数,初始rem[x]=size[x],特别的rem[a[1]]=n(a[1]是虚树最高节点,n是原树总点数,这样是为了保存a[1]往上延伸的节点)

每次处理完边(x,y),rem[x]-=size[z]。

全部做完后,对虚树的每个点f[belong[x]]+=rem[x]。

#include<cstdio>
#include<cctype>
#include<cstring>
#include<algorithm>
using namespace std;
int read(){
int s=,t=;char c;
while(!isdigit(c=getchar()))if(c=='-')t=-;
do{s=s*+c-'';}while(isdigit(c=getchar()));
return s*t;
}
const int maxn=;
int n,N,size[maxn],in[maxn],ou[maxn],f[maxn],a[maxn*],b[maxn];
int deep[maxn],tot,first[maxn],belong[maxn],st[maxn];
int rem[maxn];
bool v[maxn];
namespace cyc{
int first[maxn],tot,dfsnum=,f[maxn][];
struct edge{int v,from;}e[maxn*];
void insert(int u,int v){tot++;e[tot].v=v;e[tot].from=first[u];first[u]=tot;}
void dfs(int x,int fa){
in[x]=++dfsnum;size[x]=;
for(int j=;(<<j)<=deep[x];j++)f[x][j]=f[f[x][j-]][j-];
for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa){
deep[e[i].v]=deep[x]+;
f[e[i].v][]=x;
dfs(e[i].v,x);
size[x]+=size[e[i].v];
}
ou[x]=dfsnum;
}
int lca(int x,int y){
if(deep[x]<deep[y])swap(x,y);
int d=deep[x]-deep[y];
for(int j=;j<=;j++)if(d&(<<j))x=f[x][j];
if(x==y)return x;
for(int j=;j>=;j--)if((<<j)<=deep[x]&&f[x][j]!=f[y][j]){
x=f[x][j];y=f[y][j];
}
return f[x][];
}
void build(){
n=read();
for(int i=;i<n;i++){
int u=read(),v=read();
insert(u,v);insert(v,u);
}
dfs(,-);
}
}
int D(int x,int y){if(!y)return 0x3f3f3f3f;else return deep[x]+deep[y]-*deep[cyc::lca(x,y)];}
bool cmp(int a,int b){return in[a]<in[b];}
bool check(int x,int y){return in[x]<=in[y]&&ou[y]<=ou[x];}
struct edge{int u,v,from;}e[maxn*];
void insert(int u,int v){tot++;e[tot].u=u;e[tot].v=v;e[tot].from=first[u];first[u]=tot;}
void dfs1(int x){
if(v[x])belong[x]=x;
for(int i=first[x];i;i=e[i].from){
dfs1(e[i].v);
int A=D(x,belong[x]),B=D(x,belong[e[i].v]);
if(B<A||(B==A&&belong[e[i].v]<belong[x]))belong[x]=belong[e[i].v];
}
}
void dfs2(int x){
for(int i=first[x];i;i=e[i].from){
int A=D(e[i].v,belong[x]),B=D(e[i].v,belong[e[i].v]);
if(A<B||(A==B&&belong[x]<belong[e[i].v]))belong[e[i].v]=belong[x];
dfs2(e[i].v);
}
}
void solve(int x,int y){
int z=y;
for(int i=;i>=;i--)if((<<i)<=deep[z]&&deep[cyc::f[z][i]]>=deep[x]+)z=cyc::f[z][i];
rem[x]-=size[z];
if(belong[x]==belong[y]){f[belong[x]]+=size[z]-size[y];return;}//
int m=y;
for(int i=;i>=;i--)if((<<i)<=deep[m]){
if(deep[cyc::f[m][i]]<deep[x])continue;
int A=D(cyc::f[m][i],belong[x]),B=D(cyc::f[m][i],belong[y]);
if(B<A||(A==B&&belong[x]>belong[y]))m=cyc::f[m][i];//
}
f[belong[x]]+=size[z]-size[m];
f[belong[y]]+=size[m]-size[y];
}
void work(){
int last=read();N=last;
for(int i=;i<=N;i++)a[i]=read(),f[b[i]=a[i]]=,v[a[i]]=;
sort(a+,a+N+,cmp);
for(int i=;i<last;i++)a[++N]=cyc::lca(a[i],a[i+]);
sort(a+,a+N+,cmp);
N=unique(a+,a+N+)-a-;
tot=;
for(int i=;i<=N;i++)first[a[i]]=,belong[a[i]]=,rem[a[i]]=size[a[i]];
rem[a[]]=n;
int top=;
for(int i=;i<=N;i++){
while(top&&!check(st[top],a[i]))top--;
if(top)insert(st[top],a[i]);
st[++top]=a[i];
}
dfs1(a[]);dfs2(a[]);
for(int i=;i<=tot;i++)solve(e[i].u,e[i].v);
for(int i=;i<=N;i++)f[belong[a[i]]]+=rem[a[i]];
for(int i=;i<=last;i++)printf("%d ",f[b[i]]),v[b[i]]=;
}
int main(){
cyc::build();
int m=read();
while(m--)work();
return ;
}

(倍增求LCA)

注意,由于多次调用LCA,所以用倍增LCA算法有很大的常数,甚至在LOJ会导致TLE。

但是倍增求LCA会使过程更简洁。

下面这份代码是树链剖分求LCA,代码可读性不高。

#include<cstdio>
#include<cctype>
#include<cstring>
#include<algorithm>
using namespace std;
int read(){
int s=,t=;char c;
while(!isdigit(c=getchar()))if(c=='-')t=-;
do{s=s*+c-'';}while(isdigit(c=getchar()));
return s*t;
}
const int maxn=;
int n,N,size[maxn],in[maxn],ou[maxn],f[maxn],a[maxn*],b[maxn];
int deep[maxn],tot,first[maxn],belong[maxn],st[maxn];
int rem[maxn];
bool v[maxn];
namespace cyc{
int first[maxn],tot,dfsnum=,f[maxn][];
struct edge{int v,from;}e[maxn*];
void insert(int u,int v){tot++;e[tot].v=v;e[tot].from=first[u];first[u]=tot;}
void dfs(int x,int fa){
in[x]=++dfsnum;size[x]=;
for(int j=;(<<j)<=deep[x];j++)f[x][j]=f[f[x][j-]][j-];
for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa){
deep[e[i].v]=deep[x]+;
f[e[i].v][]=x;
dfs(e[i].v,x);
size[x]+=size[e[i].v];
}
ou[x]=dfsnum;
}
int lca(int x,int y){
if(deep[x]<deep[y])swap(x,y);
int d=deep[x]-deep[y];
for(int j=;j<=;j++)if(d&(<<j))x=f[x][j];
if(x==y)return x;
for(int j=;j>=;j--)if((<<j)<=deep[x]&&f[x][j]!=f[y][j]){
x=f[x][j];y=f[y][j];
}
return f[x][];
}
void build(){
n=read();
for(int i=;i<n;i++){
int u=read(),v=read();
insert(u,v);insert(v,u);
}
dfs(,-);
}
}
int D(int x,int y){if(!y)return 0x3f3f3f3f;else return deep[x]+deep[y]-*deep[cyc::lca(x,y)];}
bool cmp(int a,int b){return in[a]<in[b];}
bool check(int x,int y){return in[x]<=in[y]&&ou[y]<=ou[x];}
struct edge{int u,v,from;}e[maxn*];
void insert(int u,int v){tot++;e[tot].u=u;e[tot].v=v;e[tot].from=first[u];first[u]=tot;}
void dfs1(int x){
if(v[x])belong[x]=x;
for(int i=first[x];i;i=e[i].from){
dfs1(e[i].v);
int A=D(x,belong[x]),B=D(x,belong[e[i].v]);
if(B<A||(B==A&&belong[e[i].v]<belong[x]))belong[x]=belong[e[i].v];
}
}
void dfs2(int x){
for(int i=first[x];i;i=e[i].from){
int A=D(e[i].v,belong[x]),B=D(e[i].v,belong[e[i].v]);
if(A<B||(A==B&&belong[x]<belong[e[i].v]))belong[e[i].v]=belong[x];
dfs2(e[i].v);
}
}
void solve(int x,int y){
int z=y;
for(int i=;i>=;i--)if((<<i)<=deep[z]&&deep[cyc::f[z][i]]>=deep[x]+)z=cyc::f[z][i];
rem[x]-=size[z];
if(belong[x]==belong[y]){f[belong[x]]+=size[z]-size[y];return;}//
int m=y;
for(int i=;i>=;i--)if((<<i)<=deep[m]){
if(deep[cyc::f[m][i]]<deep[x])continue;
int A=D(cyc::f[m][i],belong[x]),B=D(cyc::f[m][i],belong[y]);
if(B<A||(A==B&&belong[x]>belong[y]))m=cyc::f[m][i];//
}
f[belong[x]]+=size[z]-size[m];
f[belong[y]]+=size[m]-size[y];
}
void work(){
int last=read();N=last;
for(int i=;i<=N;i++)a[i]=read(),f[b[i]=a[i]]=,v[a[i]]=;
sort(a+,a+N+,cmp);
for(int i=;i<last;i++)a[++N]=cyc::lca(a[i],a[i+]);
sort(a+,a+N+,cmp);
N=unique(a+,a+N+)-a-;
tot=;
for(int i=;i<=N;i++)first[a[i]]=,belong[a[i]]=,rem[a[i]]=size[a[i]];
rem[a[]]=n;
int top=;
for(int i=;i<=N;i++){
while(top&&!check(st[top],a[i]))top--;
if(top)insert(st[top],a[i]);
st[++top]=a[i];
}
dfs1(a[]);dfs2(a[]);
for(int i=;i<=tot;i++)solve(e[i].u,e[i].v);
for(int i=;i<=N;i++)f[belong[a[i]]]+=rem[a[i]];
for(int i=;i<=last;i++)printf("%d ",f[b[i]]),v[b[i]]=;
}
int main(){
cyc::build();
int m=read();
while(m--)work();
return ;
}

upd:复习一下。

【建虚树】

第一步、所有关键点按DFS序排序,加入LCA后排序去重。

第二步、按顺序用栈维护构造出虚树。

第三步、虚树DP后清零。

【此题的虚树DP】

第一步、处理虚树上点与点。

第二步、处理虚树边上点的子树。

第三步、处理虚树点的 [ 不在虚树上的孩子 ] 子树。(通过减虚树孩子的sz得到)

【BZOJ】3572: [Hnoi2014]世界树 虚树+倍增的更多相关文章

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

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

  2. BZOJ 3572: [Hnoi2014]世界树 [虚树 DP 倍增]

    传送门 题意: 一棵树,多次询问,给出$m$个点,求有几个点到给定点最近 写了一晚上... 当然要建虚树了,但是怎么$DP$啊 大爷题解传送门 我们先求出到虚树上某个点最近的关键点 然后枚举所有的边$ ...

  3. bzoj 3572 [Hnoi2014]世界树——虚树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3572 关于虚树:https://www.cnblogs.com/zzqsblog/p/556 ...

  4. BZOJ 3572: [Hnoi2014]世界树 虚树 树形dp

    https://www.lydsy.com/JudgeOnline/problem.php?id=3572 http://hzwer.com/6804.html 写的时候参考了hzwer的代码,不会写 ...

  5. BZOJ 3572 [HNOI2014]世界树 (虚树+DP)

    题面:BZOJ传送门 洛谷传送门 题目大意:略 细节贼多的虚树$DP$ 先考虑只有一次询问的情况 一个节点$x$可能被它子树内的一个到x距离最小的特殊点管辖,还可能被管辖fa[x]的特殊点管辖 跑两次 ...

  6. bzoj 3572: [Hnoi2014]世界树 虚树

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

  7. BZOJ 3572: [Hnoi2014]世界树

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

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

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

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

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

随机推荐

  1. erlang随机排列数组

    参考karl's answer 1> L = lists:seq(1,10). [1,2,3,4,5,6,7,8,9,10] Associate a random number R with e ...

  2. 201621123037 《Java程序设计》第8周学习总结

    作业08-集合 1. 本周学习总结 以你喜欢的方式(思维导图或其他)归纳总结集合相关内容. 答: 思维导图: 其他-笔记: 2. 书面作业 1. ArrayList代码分析 1.1 解释ArrayLi ...

  3. Tomcat指定JDK路径

    一.应用实例 一般情况下一台服务器只跑一个业务,那么就直接配置一套环境,设置好Java环境变量即可.某些时候一台服务器上会安装多个业务,而且各个业务需要的JDK版本各不相同,或者为了使业务独立开来,需 ...

  4. Spring 学习 5- task 定时任务

    Spring-Task 1.这是网上的: 后面是我自己的配置 Spring3.0以后自主开发的定时任务工具,spring task,可以将它比作一个轻量级的Quartz,而且使用起来很简单,除spri ...

  5. ubuntu下搭建openGL环境

    1.      建立基本编译环境 sudo apt-get install build-essential 2.      安装OpenGL Library sudo apt-get install ...

  6. BZOJ4922 Karp-de-Chant Number(贪心+动态规划)

    首先将每个括号序列转化为三元组(ai,bi,ci),其中ai为左括号-右括号数量,bi为前缀最小左括号-右括号数,ci为序列长度.问题变为在满足Σai=0,bi+Σaj>=0 (j<i)的 ...

  7. java中初始化块、静态初始化块和构造方法

    (所谓的初始化方法init()是另一回事, 在构造方法之后执行, 注意不要混淆) 在Java中,有两种初始化块:静态初始化块和非静态初始化块.它们都是定义在类中,用大括号{}括起来,静态代码块在大括号 ...

  8. 🔺Count on a tree SPOJ - COT (无能为力。。。)

    https://cn.vjudge.net/problem/SPOJ-COT 插上 大佬的代码 和 我的...以后再看吧... Count on a tree 大佬:http://www.cnblog ...

  9. Astronauts UVALive - 3713(2-SAT)

    大白书例题 #include <iostream> #include <cstdio> #include <sstream> #include <cstrin ...

  10. 【洛谷】CYJian的水题大赛【第二弹】解题报告

    点此进入比赛 T1: JerryC Loves Driving 第一题应该就是一道水分题(然而我只水了130分),我的主要做法就是暴力模拟,再做一些小小的优化(蠢得我自己都不想说了). My Code ...