BZOJ 3572: [Hnoi2014]世界树 [虚树 DP 倍增]
题意:
一棵树,多次询问,给出$m$个点,求有几个点到给定点最近
写了一晚上...
当然要建虚树了,但是怎么$DP$啊
我们先求出到虚树上某个点最近的关键点
然后枚举所有的边$(f,x)$,讨论一下边上的点的子树应该靠谁更近
倍增求出分界点
注意有些没出现在虚树上的子树
注意讨论的时候只讨论链上的不包括端点,否则$f$的子树会被贡献多次
学到的一些$trick:$
1.$pair$的妙用
2.不需要建出虚树只要求虚树的$dfs$序(拓扑序)和$fa$就可以$DP$了
注意$DP$的时候必须先用儿子更新父亲再用父亲更新儿子,因为父亲的最优值有可能在其他儿子
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
#define pii pair<int,int>
#define MP make_pair
#define fir first
#define sec second
typedef long long ll;
const int N=3e5+,INF=1e9;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
} int n,Q;
struct Edge{
int v,ne,w;
}e[N<<];
int cnt,h[N];
inline void ins(int u,int v){
cnt++;
e[cnt].v=v;e[cnt].ne=h[u];h[u]=cnt;
cnt++;
e[cnt].v=u;e[cnt].ne=h[v];h[v]=cnt;
}
int fa[N][],deep[N],dfn[N],dfc,size[N],All;
void dfs(int u){
dfn[u]=++dfc;
size[u]=;
for(int i=;(<<i)<=deep[u];i++)
fa[u][i]=fa[ fa[u][i-] ][i-];
for(int i=h[u];i;i=e[i].ne)
if(e[i].v!=fa[u][]){
fa[e[i].v][]=u;
deep[e[i].v]=deep[u]+;
dfs(e[i].v);
size[u]+=size[e[i].v];
}
}
inline int lca(int x,int y){
if(deep[x]<deep[y]) swap(x,y);
int bin=deep[x]-deep[y];
for(int i=;i>=;i--)
if((<<i)&bin) x=fa[x][i];
for(int i=;i>=;i--)
if(fa[x][i]!=fa[y][i]) x=fa[x][i],y=fa[y][i];
return x==y ? x :fa[x][];
} int a[N],st[N],par[N],dis[N],t[N],m,ans[N];
int remain[N];
inline bool cmp(int x,int y){return dfn[x]<dfn[y];}
inline void ins2(int x,int y){par[y]=x;dis[y]=deep[y]-deep[x];}
pii g[N];
void dp(int m){
for(int i=m;i>;i--){
int x=t[i],f=par[x];
g[f]=min(g[f],MP(g[x].fir+dis[x],g[x].sec));
}
for(int i=;i<=m;i++){
int x=t[i],f=par[x];
g[x]=min(g[x],MP(g[f].fir+dis[x],g[f].sec));
}
}
inline int jump1(int x,int tar){
for(int i=;i>=;i--)
if(deep[ fa[x][i] ]>=tar) x=fa[x][i];
return x;
}
inline int jump(int x,int tar){
int bin=deep[x]-tar;
for(int i=;i>=;i--)
if((<<i)&bin) x=fa[x][i];
return x;
}
int ora[N];
void solve(){
int n=read(),m=;
for(int i=;i<=n;i++)
ora[i]=a[i]=read(),t[++m]=a[i],g[a[i]]=MP(,a[i]);
sort(a+,a++n,cmp); int top=;
for(int i=;i<=n;i++){
if(!top) {st[++top]=a[i];continue;}
int x=a[i],f=lca(x,st[top]);
while(dfn[f]<dfn[st[top]]){
if(dfn[f]>=dfn[st[top-]]){
ins2(f,st[top--]);
if(f!=st[top]) st[++top]=f,t[++m]=f,g[f]=MP(INF,);
break;
}else ins2(st[top-],st[top]),top--;
}
st[++top]=x;
}
while(top>) ins2(st[top-],st[top]),top--; sort(t+,t++m,cmp);
dp(m);
for(int i=;i<=m;i++) remain[t[i]]=size[t[i]]; ans[ g[t[]].sec ]+=All-size[t[]];
for(int i=;i<=m;i++){
int x=t[i],f=par[x];par[x]=;
int t=jump(x,deep[f]+);
remain[f]-=size[t];
if(g[x].sec == g[f].sec) ans[ g[x].sec ]+=size[t]-size[x];
else{
int len=g[x].fir + g[f].fir + dis[x], mid=deep[x]-(len/-g[x].fir);
if( !(len&) && g[f].sec<g[x].sec ) mid++;
int y=jump(x,mid);
ans[ g[f].sec ]+=size[t]-size[y];
ans[ g[x].sec ]+=size[y]-size[x];
}
}
for(int i=;i<=m;i++) ans[ g[t[i]].sec ]+=remain[t[i]];
for(int i=;i<=n;i++) printf("%d%c",ans[ora[i]],i==n?'\n':' '),ans[ora[i]]=;
}
int main(){
//freopen("in","r",stdin);
n=read();All=n;
for(int i=;i<n;i++) ins(read(),read());
dfs();
Q=read();
while(Q--) solve();
}
BZOJ 3572: [Hnoi2014]世界树 [虚树 DP 倍增]的更多相关文章
- BZOJ 3572 [HNOI2014]世界树 (虚树+DP)
题面:BZOJ传送门 洛谷传送门 题目大意:略 细节贼多的虚树$DP$ 先考虑只有一次询问的情况 一个节点$x$可能被它子树内的一个到x距离最小的特殊点管辖,还可能被管辖fa[x]的特殊点管辖 跑两次 ...
- bzoj 3572: [Hnoi2014]世界树 虚树 && AC500
3572: [Hnoi2014]世界树 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 520 Solved: 300[Submit][Status] ...
- BZOJ 3572: [Hnoi2014]世界树 虚树 树形dp
https://www.lydsy.com/JudgeOnline/problem.php?id=3572 http://hzwer.com/6804.html 写的时候参考了hzwer的代码,不会写 ...
- bzoj3572[Hnoi2014] 世界树 虚树+dp+倍增
[Hnoi2014]世界树 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1921 Solved: 1019[Submit][Status][Dis ...
- bzoj 3572 [Hnoi2014]世界树——虚树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3572 关于虚树:https://www.cnblogs.com/zzqsblog/p/556 ...
- bzoj 3572: [Hnoi2014]世界树 虚树
题目: Description 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森,在他们的信条里,公平是使世界树能够生生 ...
- 【BZOJ】3572: [Hnoi2014]世界树 虚树+倍增
[题意]给定n个点的树,m次询问,每次给定ki个特殊点,一个点会被最近的特殊点控制,询问每个特殊点控制多少点.n,m,Σki<=300000. [算法]虚树+倍增 [题解]★参考:thy_asd ...
- 洛谷 P3233 [HNOI2014]世界树(虚树+dp)
题面 luogu 题解 数据范围已经告诉我们是虚树了,考虑如何在虚树上面\(dp\) 以下摘自hzwer博客: 构建虚树以后两遍dp处理出虚树上每个点最近的议事处 然后枚举虚树上每一条边,考虑其对两端 ...
- BZOJ 3572: [Hnoi2014]世界树
BZOJ 3572: [Hnoi2014]世界树 标签(空格分隔): OI-BZOJ OI-虚数 OI-树形dp OI-倍增 Time Limit: 20 Sec Memory Limit: 512 ...
随机推荐
- 搬寝室(经典dp)
题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=1421 hdu_1421:搬寝室 Time Limit: 2000/1000 MS (Java/Othe ...
- FtpUtil.java测试 (淘淘商城第三课文件上传)
首先在common-taotao中创建一个utils包,复制FtpUtil.java到其中.然后如下: @Test public void testFtpUtil() throws Exception ...
- Vue-cli搭建完项目,各文件解释
脚手架vue-cli搭建完成后,会生成一些文件,总结学习一下这些文件是做什么用的:1.一级目录: build和config文件夹是wbepack配置的文件夹: node_modules是在我npm i ...
- JXLS 2.4.0系列教程(三)——嵌套循环是怎么做到的
注:本文代码在第一篇文章基础上修改而成,请务必先阅读第一篇文章. http://www.cnblogs.com/foxlee1024/p/7616987.html 本文也不会过多的讲解模板中遍历表达式 ...
- [sklearn]官方例程-Imputing missing values before building an estimator 随机填充缺失值
官方链接:http://scikit-learn.org/dev/auto_examples/plot_missing_values.html#sphx-glr-auto-examples-plot- ...
- android 基础03 -- Intent
Android 中的 Intent 是将要执行的操作的一种抽象的描述,是一个用于Android 各个组件之间传递消息的对象. Intent 的基本用法 Intent 基本的使用方法主要有三种: 启动一 ...
- WKWebView--JS调用OC的方法
WKWebView---JS调用OC方法 一.使用的协议进行简单的介绍 1.在WKWebView中OC和JS交互也非常简单,WebKit的库中有个代理WKScriptMessageHandler就是专 ...
- python3 第十章 - 如何进行进制转化
在计算机的世界里,2进制是主流,而在人类的自然世界中,10进制是主流,那么在这之间必然就会存在进制转化的问题.本章我们就来谈谈进制转化,也希望通过本章加深您对前些章所学知识的理解. 原理:先说说关于位 ...
- 腾讯工程师带你深入解析 MySQL binlog
欢迎大家前往云+社区,获取更多腾讯海量技术实践干货哦~ 本文由 腾讯云数据库内核团队 发布在云+社区 1.概述 binlog是Mysql sever层维护的一种二进制日志,与innodb引擎中的red ...
- jquery与js添加子元素
例如在select中添加option JQuery做法: <select id="myselect" name="myselect"> </s ...