BZOJ 3611 大工程 (虚树)
题面
国家有一个大工程,要给一个非常大的交通网络里建一些新的通道。
我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上。
在 2 个国家 a,b 之间建一条新通道需要的代价为树上 a,b 的最短路径。
现在国家有很多个计划,每个计划都是这样,我们选中了 k 个点,然后在它们两两之间 新建 C(k,2)条 新通道。
现在对于每个计划,我们想知道:
1.这些新通道的代价和
2.这些新通道中代价最小的是多少
3.这些新通道中代价最大的是多少
分析
显然需要建虚树,我们考虑如何在虚树上DP(注意:以下定义的点和子树都是虚树上的)
设sz[x]表示x的子树内有多少个询问点,sum[x]表示x子树内的路径长度和,dmin[x]表示x子树内到x的最小路径长度,dmax[x]表示 x子树内到x的最大路径长度
对于x的子节点y,我们可以写出如下的状态转移方程
sum[x]+=sum[y]+(k-sz[y]) · sz[y] ·dist(x,y)
x的子树内的路径分为两种:一种是在y的子树内部的路径,一种是从y的子树内的sz[y]个节点到子树外(k-sz[y])个节点)
dmin[x]=min(dmin[x],dmin[y]+dist(x,y));
dmax[x]=max(dmax[x],dmax[y]+dist(x,y));
long long siz[maxn];//siz[x],x子树内询问点的个数
long long sum[maxn],dmin[maxn],dmax[maxn];
//sum[x] x子树内的路径长度和
//dmin[x] x子树内到x的最小路径长度
//dmax[x] x子树内到x的最大路径长度
long long ans_sum,ans_min,ans_max;
void dfs2(int x,int fa){
siz[x]=is_q[x];
dmax[x]=0;
dmin[x]=INF;
sum[x]=0;
for(int i=T2.head[x];i;i=T2.E[i].next){
int y=T2.E[i].to;
long long len=deep[y]-deep[x];
if(y!=fa){
dfs2(y,x);
sum[x]+=sum[y]+siz[y]*(k-siz[y])*len;
//y子树内部路径 + y子树到外面其他询问点的路径
siz[x]+=siz[y];
ans_min=min(ans_min,dmin[x]+dmin[y]+len);
ans_max=max(ans_max,dmax[x]+dmax[y]+len);
dmin[x]=min(dmin[x],dmin[y]+len);
dmax[x]=max(dmax[x],dmax[y]+len);
}
}
if(is_q[x]){
ans_min=min(ans_min,dmin[x]);
ans_max=max(ans_max,dmax[x]);
dmin[x]=0;//当前最小值为0,这样祖先节点就可以用它来更新最小值(最小路径可能是祖先~x)
}
T2.head[x]=0;
}
注意,此题需要卡常
可以预处理出2的i次方,然后在lca里面优化倍增,详情见代码
代码
//http://119.29.55.79/problem/276
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define maxn 1000005
#define maxlog 32
#define INF 0x3f3f3f3f3f3f3f3f
using namespace std;
inline void qread(int &x){
x=0;
int sign=1;
char c=getchar();
while(c<'0'||c>'9'){
if(c=='-') sign=-1;
c=getchar();
}
while(c>='0'&&c<='9'){
x=x*10+c-'0';
c=getchar();
}
x=x*sign;
}
inline void qprint(long long x){
if(x<0){
putchar('-');
qprint(-x);
}else if(x==0){
putchar('0');
return;
}else{
if(x/10>0) qprint(x/10);
putchar('0'+x%10);
}
}
int n,m;
struct graph{
struct edge{
int from;
int to;
int next;
}E[maxn<<1];
int head[maxn];
int sz;
void add_edge(int u,int v){
sz++;
E[sz].from=u;
E[sz].to=v;
E[sz].next=head[u];
head[u]=sz;
sz++;
E[sz].from=v;
E[sz].to=u;
E[sz].next=head[v];
head[v]=sz;
}
graph(){
sz=1;
}
}T1,T2;
int log2n;
int tim=0;
int dfn[maxn];
int deep[maxn];
int bin[maxn];//卡常用
int anc[maxn][maxlog];
void dfs1(int x,int fa){
dfn[x]=++tim;
deep[x]=deep[fa]+1;
anc[x][0]=fa;
for(int i=1;bin[i]<=deep[x];i++)anc[x][i]=anc[anc[x][i-1]][i-1];
for(int i=T1.head[x];i;i=T1.E[i].next){
int y=T1.E[i].to;
if(y!=fa){
dfs1(y,x);
}
}
}
int lca(int x,int y){
if(deep[x]<deep[y]) swap(x,y);
int t=deep[x]-deep[y];
for(int i=0;bin[i]<=t;i++){
if(bin[i]&t){
x=anc[x][i];
}
}
if(x==y) return x;
for(int i=log2n;i>=0;i--){
if(anc[x][i]!=anc[y][i]){
x=anc[x][i];
y=anc[y][i];
}
}
return anc[x][0];
}
int k;
int top=0;
int s[maxn];
int in[maxn];
int cmp(int x,int y){
return dfn[x]<dfn[y];
}
int is_q[maxn];//是否是询问点
void insert(int x){
if(top<=1){
s[++top]=x;
return;
}
int lc=lca(x,s[top]);
if(lc==s[top]){
s[++top]=x;
return;
}
while(top>1&&deep[s[top-1]]>=deep[lc]){
T2.add_edge(s[top-1],s[top]);
// printf("vtree : %d->%d\n",s[top-1],s[top]);
top--;
}
if(s[top]!=lc){
T2.add_edge(lc,s[top]);
// printf("vtree : %d->%d\n",lc,s[top]);
}
s[top]=lc;
s[++top]=x;
}
long long siz[maxn];//siz[x],x子树内询问点的个数
long long sum[maxn],dmin[maxn],dmax[maxn];
//sum[x] x子树内的路径长度和
//dmin[x] x子树内到x的最小路径长度
//dmax[x] x子树内到x的最大路径长度
long long ans_sum,ans_min,ans_max;
void dfs2(int x,int fa){
siz[x]=is_q[x];
dmax[x]=0;
dmin[x]=INF;
sum[x]=0;
for(int i=T2.head[x];i;i=T2.E[i].next){
int y=T2.E[i].to;
long long len=deep[y]-deep[x];
if(y!=fa){
dfs2(y,x);
sum[x]+=sum[y]+siz[y]*(k-siz[y])*len;
//y子树内部路径 + y子树到外面其他询问点的路径
siz[x]+=siz[y];
ans_min=min(ans_min,dmin[x]+dmin[y]+len);
ans_max=max(ans_max,dmax[x]+dmax[y]+len);
dmin[x]=min(dmin[x],dmin[y]+len);
dmax[x]=max(dmax[x],dmax[y]+len);
}
}
if(is_q[x]){
ans_min=min(ans_min,dmin[x]);
ans_max=max(ans_max,dmax[x]);
dmin[x]=0;//当前最小值为0,这样祖先节点就可以用它来更新最小值(最小路径可能是祖先~x)
}
T2.head[x]=0;
}
void solve(){
sort(in+1,in+1+k,cmp);
for(int i=1;i<=k;i++) is_q[in[i]]=1;
int root=lca(in[1],in[2]);
for(int i=3;i<=k;i++){
root=lca(root,in[i]);
}
top=0;
s[++top]=root;
for(int i=1;i<=k;i++){
if(in[i]!=root)insert(in[i]);
}
while(top>1){
T2.add_edge(s[top-1],s[top]);
// printf("vtree : %d->%d\n",s[top-1],s[top]);
top--;
}
ans_sum=0;
ans_max=0;
ans_min=INF;
dfs2(root,0);
ans_sum=sum[root];
for(int i=1;i<=k;i++) is_q[in[i]]=0;
T2.sz=1;
}
int main(){
#ifdef FILE_IO
freopen("tree9.in","r",stdin);
freopen("tree9.ans","w",stdout);
#endif
int u,v;
qread(n);
log2n=log2(n)+1;
bin[0]=1;
for(int i=1;i<=log2n;i++) bin[i]=bin[i-1]*2;
for(int i=1;i<n;i++){
qread(u);
qread(v);
T1.add_edge(u,v);
}
dfs1(1,0);
qread(m);
for(int i=1;i<=m;i++){
qread(k);
for(int j=1;j<=k;j++){
qread(in[j]);
}
solve();
qprint(ans_sum);putchar(' ');
qprint(ans_min);putchar(' ');
qprint(ans_max);putchar('\n');
}
}
BZOJ 3611 大工程 (虚树)的更多相关文章
- bzoj 3611: [Heoi2014]大工程 虚树
题目: 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上. 在 2 个国家 a,b 之间建一条新通道需要的代价为树上 ...
- bzoj 3611(洛谷 4103) [Heoi2014]大工程——虚树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3611 https://www.luogu.org/problemnew/show/P4103 ...
- BZOJ.3611.[HEOI2014]大工程(虚树 树形DP)
题目链接 要求的和.最大值.最小值好像都可以通过O(n)的树形DP做,总询问点数<=2n. 于是建虚树就可以了.具体DP见DP()函数,维护三个值sum[],mx[],mn[]. sum[]要开 ...
- BZOJ 3611 [Heoi2014]大工程 ——虚树
虚树第二题.... 同BZOJ2286 #include <map> #include <cmath> #include <queue> #include < ...
- luogu P4103 [HEOI2014]大工程 虚树 + 树形 DP
Description 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上. 在 2 个国家 a,b 之间建一条新通 ...
- 【HEOI2014】大工程<虚树>
虚树 我们每天都用心思索着,这究竟是为了什么呢?我想我也不知道,只是觉得如果人不思考问题就很无聊. 我觉得虚树不是什么数据结构,就是一种技巧或者工具.它能把树中\(k\)个关键点以\(O(klogk) ...
- 洛谷P4103 [HEOI2014]大工程(虚树 树形dp)
题意 链接 Sol 虚树. 首先建出虚树,然后直接树形dp就行了. 最大最小值直接维护子树内到该节点的最大值,然后合并两棵子树的时候更新一下答案. 任意两点的路径和可以考虑每条边两边的贡献,\(d[x ...
- [HEOI2014][bzoj3611] 大工程 [虚树+dp]
题面: 传送门 思路: 又是一道虚树入门级的题目,但是这道题的实际难点在于dp 首先,这道题是可以点分治做的,而且因为6s时限随便浪,所以写点分治也不是不可以 但是,dp因为$O\left(n\rig ...
- bzoj 3611[Heoi2014]大工程 虚树+dp
题意: 给一棵树 每次选 k 个关键点,然后在它们两两之间 新建 C(k,2)条 新通道. 求: 1.这些新通道的代价和 2.这些新通道中代价最小的是多少 3.这些新通道中代价最大的是多少 分析:较常 ...
随机推荐
- 分析abex'crackme#1
测试文件下载:https://www.wocloud.com.cn/webclient/share/sindex.action?id=i9K_Br6TgE7Kf_YTF04yHmKcRy5TUdZ8U ...
- 剑指offer学习--初级c++面试题
定义一个空的类型,里面没有任何成员函数和成员变量,对该类型求sizeof,得到的结果是多少? 答案是1.空类型中的实例中不包含任何信息,本来求sizeof应该是0,但是当我们声明该类型的实例的时候,他 ...
- [BJOI2006]狼抓兔子(网络流)
题目描述 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的,而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的地形: ...
- springboot打包成jar文件无法正常运行,解决办法已经找到
1.用intellij idea 创建了一个springboot的项目,前期都运行的好好的,在ide中可以正常运行,但是打包成Jar运行却一直报错. 2.经过不懈探索,终于找到解决办法 3.首先,找到 ...
- Ubuntu18.04 安装 Idea 2018.2
https://blog.csdn.net/weixx3/article/details/81136822 Ubuntu18.04 安装 Idea 2018.2环境信息:OS:Ubuntu18.04J ...
- docker:docker的基本了解
1.什么是docker 简单的理解:docker相当于vmvare,容器相当于多个虚拟机,vmvare上可以运行ubantu16.04的虚拟机,也可以运行centos虚拟机,还可以运行redhat虚拟 ...
- python 获取文件行数
#如果要统计文件的行数,可以这样写: count = len(open(filepath, 'r').readlines()) #这种方法简单,但是可能比较慢,当文件比较大时甚至不能工作. #可以利用 ...
- Testng的使用总结(内容待持续更新)
testng 6.8使用多线程时,在pom的surefire插件始终无效 -->升级testng版本,在6.8版本中无任何提示的 如何调用自定义的报告的 -->在testng中,有个IRe ...
- mobx学习笔记02——mobx基础语法(class)
新的语法可能不被浏览器支持,可以使用babel转换为浏览器支持的代码格式: 为什么要定义class? js是一门面向对象的编程语言.需要利用类来复用代码,提高编程效率. 需要什么样的class能力? ...
- fat文件系统
在主引导区后面就是FAT表.从上面可以得知一个FAT表是229个扇区.它里边的内容很简单,里边的内容就是指出下一个簇在哪里.你的盘有多少个簇,那么它的FAT表就要有多少个项来描述它们.因为FAT16是 ...