luoguP3233 [HNOI2014]世界树
题意
看见数据范围就知道是虚树,于是先建出虚树。
考虑先求出虚树上的点的管理点,显然两边dfs,一遍从下往上,一遍从上往下。
之后考虑不在虚树上的点,对于虚树上的每一条边\((u,v)\),我们考虑上面未出现在虚树上的点的个数是多少,显然是\(u\)包含\(v\)的儿子\(x\)的子树的大小减去\(v\)子树的大小,这个可以倍增求出。
考虑\(u,v\)两点的管理点是否相等,如果相等那么直接加上\(size[x]-size[v]\)即可,不相等\(u->v\)就必定存在一个分界点,于是我们可以倍增求出。
这时还会有考虑不到的点,用个数组\(g[x]\)记下x的子树中有多少点未被分配,之后并到\(x\)上。
code:
#include<bits/stdc++.h>
using namespace std;
const int maxn=300010;
int n,m,cnt,tim,top,t,num;
int head[maxn],a[maxn],sta[maxn],dfn[maxn],dep[maxn],ans[maxn],size[maxn],b[maxn],c[maxn],col[maxn],g[maxn];
int f[maxn][20];
struct edge{int to,nxt;}e[maxn<<1];
inline bool cmp(int x,int y){return dfn[x]<dfn[y];}
inline void add(int u,int v)
{
e[++cnt].nxt=head[u];
head[u]=cnt;
e[cnt].to=v;
}
void dfs_pre(int x,int fa)
{
dep[x]=dep[fa]+1;size[x]=1;dfn[x]=++tim;
for(int i=1;i<=t;i++)f[x][i]=f[f[x][i-1]][i-1];
for(int i=head[x];i;i=e[i].nxt)
{
int y=e[i].to;
if(y==fa)continue;
f[y][0]=x;dfs_pre(y,x);size[x]+=size[y];
}
}
inline int lca(int x,int y)
{
if(dep[x]>dep[y])swap(x,y);
for(int i=t;~i;i--)if(dep[f[y][i]]>=dep[x])y=f[y][i];
if(x==y)return x;
for(int i=t;~i;i--)if(f[x][i]!=f[y][i])x=f[x][i],y=f[y][i];
return f[x][0];
}
inline int dis(int x,int y){return dep[x]+dep[y]-2*dep[lca(x,y)];}
void dfs1(int x)
{
c[++num]=x;g[x]=size[x];
for(int i=head[x];i;i=e[i].nxt)
{
int y=e[i].to;
dfs1(y);
if(!col[y])continue;
if(!col[x]){col[x]=col[y];continue;}
int d1=dis(col[y],x),d2=dis(col[x],x);
if(d1<d2||(d1==d2&&col[y]<col[x]))col[x]=col[y];
}
}
void dfs2(int x)
{
for(int i=head[x];i;i=e[i].nxt)
{
int y=e[i].to;
if(!col[y])col[y]=col[x];
else
{
int d1=dis(col[y],y),d2=dis(col[x],y);
if(d1>d2||(d1==d2&&col[x]<col[y]))col[y]=col[x];
}
dfs2(y);
}
}
inline void work(int x,int y)
{
int now=y;
for(int i=t;~i;i--)if(dep[now]-(1<<i)>dep[x])now=f[now][i];
g[x]-=size[now];
if(col[x]==col[y]){ans[col[x]]+=size[now]-size[y];return;}
int tmp=y;
for(int i=t;~i;i--)
{
int nxt=f[tmp][i];
if(dep[nxt]<=dep[x])continue;
int d1=dis(nxt,col[x]),d2=dis(nxt,col[y]);
if(d1>d2||(d1==d2&&col[y]<col[x]))tmp=nxt;
}
ans[col[x]]+=size[now]-size[tmp],ans[col[y]]+=size[tmp]-size[y];
}
inline void solve()
{
num=cnt=0;
int tot;scanf("%d",&tot);
for(int i=1;i<=tot;i++)scanf("%d",&a[i]),b[i]=a[i],col[a[i]]=a[i];
sta[top=1]=1;head[1]=0;
sort(a+1,a+tot+1,cmp);
for(int i=1;i<=tot;i++)
{
if(a[i]==1)continue;
int x=lca(sta[top],a[i]);
if(x!=sta[top])
{
while(top>1&&dfn[sta[top-1]]>dfn[x])add(sta[top-1],sta[top]),top--;
if(x!=sta[top-1])head[x]=0,add(x,sta[top]),sta[top]=x;
else add(x,sta[top--]);
}
head[a[i]]=0,sta[++top]=a[i];
}
for(int i=1;i<top;i++)add(sta[i],sta[i+1]);
dfs1(1),dfs2(1);
for(int j=1;j<=num;j++)
for(int i=head[c[j]];i;i=e[i].nxt)
work(c[j],e[i].to);
for(int i=1;i<=num;i++)ans[col[c[i]]]+=g[c[i]];
for(int i=1;i<=tot;i++)printf("%d ",ans[b[i]]);
puts("");
for(int i=1;i<=num;i++)ans[c[i]]=g[c[i]]=col[c[i]]=0;
}
int main()
{
scanf("%d",&n);t=(int)log2(n)+1;
for(int i=1;i<n;i++)
{
int u,v;scanf("%d%d",&u,&v);
add(u,v),add(v,u);
}
dfs_pre(1,0);
scanf("%d",&m);
for(int i=1;i<=m;i++)solve();
return 0;
}
luoguP3233 [HNOI2014]世界树的更多相关文章
- [BZOJ3572][Hnoi2014]世界树
[BZOJ3572][Hnoi2014]世界树 试题描述 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森,在他们的信条 ...
- 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] ...
- 【BZOJ3572】[Hnoi2014]世界树 虚树
[BZOJ3572][Hnoi2014]世界树 Description 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森 ...
- bzoj3572[Hnoi2014] 世界树 虚树+dp+倍增
[Hnoi2014]世界树 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1921 Solved: 1019[Submit][Status][Dis ...
- <虚树+树型DP> HNOI2014世界树
<虚树+树型DP> HNOI2014世界树 #include <iostream> #include <cstdio> #include <cstring&g ...
- [题解] [HNOI2014] 世界树
题面 [HNOI2014]世界树 题解 从数据范围很容易看出是个虚树DP(可惜看出来了也还是不会做) 虚树大家应该都会, 不会的话自己去搜吧, 我懒得讲了, 我们在这里只需要考虑如何DP即可 首先我们 ...
- [HNOI2014]世界树
题目描述 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森,在他们的信条里,公平是使世界树能够生生不息.持续运转的根本基 ...
随机推荐
- R基础绘图
本节内容 0:小知识 1:绘图系统散点图的特征 2:基础绘图函数 3:基础绘图参数 4:图形设备 5:案例操作5个图形 0:小知识 summary() ## 对数据框或者向量进行描述性数据 read. ...
- 5.jenkins 构建任务2--PHP项目
推送代码 PHP 项目 然后我们要做的就是将 .jenkins下面 workspace下的代码. 推送到我们的项目机器上面 就ok了. 我们可以在php的jenkins的 项目配置中选择使用shell ...
- xen原理
目录:1. Xen的简介1.1 Xen的大体结构1.2 Xen对VM的称呼1.3 Xen对CPU和内存的虚拟化过程1.4 Xen对IO设备的虚拟化过程1.5 Linux Kernel对Xen的支持1. ...
- 《Spring Cloud微服务 入门 实战与进阶》
很少在周末发文,还是由于昨晚刚收到实体书,还是耐不住性子马上发文了. 一年前,耗时半年多的时间,写出了我的第一本书<Spring Cloud微服务-全栈技术与案例解析>. 时至今日,一年的 ...
- 无限可能,Elasticsearch(一)
这个世界已然被数据淹没.多年来,我们系统间流转和产生的大量数据已让我们不知所措. 现有的技术都集中在如何解决数据仓库存储以及如何结构化这些数据. 这些看上去都挺美好,直到你实际需要基于这些数据实时做决 ...
- WPF 使用XML作为绑定源时Xaml注意事项
直接在xaml定义时xml时应该注意的! xml数据 <?xml version="1.0" encoding="utf-8"?> <Stri ...
- 基于 H5 + WebGL 实现的地铁站 3D 可视化系统
前言 工业互联网,物联网,可视化等名词在我们现在信息化的大背景下已经是耳熟能详,日常生活的交通,出行,吃穿等可能都可以用信息化的方式来为我们表达,在传统的可视化监控领域,一般都是基于 Web SCAD ...
- windows 安装使用jupyter及 基础配置
jupyter 是什么Jupyter Notebooks 是一个交互式笔记本,支持运行 40 多种编程语言,它的本质是一个 开源的 Web 应用程序,我们可以将其用于创建和共享代码与文档,他可以支持实 ...
- Python - 部分PEP8规范
写代码就像写字一样,为什么有的人写的字十分漂亮,而有的人写的字过后连自己都不认识,最主要还是从一开始是否对自己严格要求.从现在开始就当自己是个初学者,把代码写漂亮点.以下截取了部分PEP8代码规范,里 ...
- Java 类加载机制(阿里)-何时初始化类
(1)阿里的面试官问了两个问题,可以不可以自己写个String类 答案:不可以,因为 根据类加载的双亲委派机制,会去加载父类,父类发现冲突了String就不再加载了; (2)能否在加载类的时候,对类的 ...