【bzoj3611】 大工程
http://www.lydsy.com/JudgeOnline/problem.php?id=3611 (题目链接)
搞了1天= =,感觉人都变蠢了。。。
题意
给出一个n个节点的树,每条边边权为1,给出q个询问,每次询问K个关键点,求出这k个点之间的两两距离和、最小距离和最大距离。
solution
构造虚树,见 http://blog.csdn.net/MashiroSky/article/details/51971718
之后在虚树上dp,有点麻烦。
用size[u]表示在以u为根的子树上的关键点个数,b[u]表示虚树上的节点u是否是关键点,f[u]表示以u为根的子树上所有关键点到u的距离之和,mn[u]表示以u为根的子树上距离u最近的关键点到u的距离,mx[u]表示以u为根的子树上距离u最远的关键点到u的距离。
代码
// bzoj3611
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<set>
#define MOD 1000000007
#define inf 2147483640
#define LL long long
#define free(a) freopen(a".in","r",stdin);freopen(a".out","w",stdout);
using namespace std;
inline LL getint() {
LL x=0,f=1;char ch=getchar();
while (ch>'9' || ch<'0') {if (ch=='-') f=-1;ch=getchar();}
while (ch>='0' && ch<='9') {x=x*10+ch-'0';ch=getchar();}
return x*f;
} const int maxn=1000010;
struct edge {int next,to,w;}e[maxn<<2];
int deep[maxn],fa[maxn][20],bin[20],dfn[maxn],head[maxn];
int mn[maxn],mx[maxn],s[maxn],a[maxn],b[maxn];
int n,K,cnt,top,q;
LL tot,ans1,ans2,f[maxn],size[maxn]; void insert(int u,int v) {
if (u==v) return;
e[++cnt].to=v;e[cnt].next=head[u];head[u]=cnt;
e[cnt].w=deep[v]-deep[u];
}
void dfs(int u) {
dfn[u]=++cnt;
for (int i=1;i<20;i++) fa[u][i]=fa[fa[u][i-1]][i-1];
for (int i=head[u];i;i=e[i].next) if (e[i].to!=fa[u][0]) {
deep[e[i].to]=deep[u]+1;
fa[e[i].to][0]=u;
dfs(e[i].to);
}
head[u]=0;
}
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 (t&bin[i]) x=fa[x][i];
for (int i=19;i>=0;i--)
if (fa[x][i]!=fa[y][i]) x=fa[x][i],y=fa[y][i];
return x==y?x:fa[x][0];
}
bool cmp(int x,int y) {
return dfn[x]<dfn[y];
}
void dp(int u) {
size[u]=b[u];
f[u]=0;
mn[u]=b[u]?0:inf;
mx[u]=b[u]?0:-inf;
for (int i=head[u];i;i=e[i].next) {
int v=e[i].to;
dp(v);
tot+=(f[u]+size[u]*e[i].w)*size[v]+f[v]*size[u];
size[u]+=size[v];
f[u]+=f[v]+e[i].w*size[v];
ans1=min(ans1,(LL)mn[u]+mn[v]+e[i].w);
ans2=max(ans2,(LL)mx[u]+mx[v]+e[i].w);
mn[u]=min(mn[u],mn[v]+e[i].w);
mx[u]=max(mx[u],mx[v]+e[i].w);
}
head[u]=0;
}
void build() {
scanf("%d",&K);
for (int i=1;i<=K;i++) scanf("%d",&a[i]),b[a[i]]=1;
sort(a+1,a+1+K,cmp);
s[++top]=1;
for (int i=1;i<=K;i++) {
int t=a[i],f=0;
while (top>0) {
f=lca(a[i],s[top]);
if (top>1 && deep[f]<deep[s[top-1]])
insert(s[top-1],s[top]),top--;
else if (deep[f]<deep[s[top]]) {insert(f,s[top--]);break;}
else break;
}
if (s[top]!=f) s[++top]=f;
s[++top]=t;
}
while (--top) insert(s[top],s[top+1]);
}
int main() {
bin[0]=1;for (int i=1;i<20;i++) bin[i]=bin[i-1]<<1;
scanf("%d",&n);
for (int i=1;i<n;i++) {
int u=getint(),v=getint();
insert(u,v);insert(v,u);
}
cnt=0;
dfs(1);
scanf("%d",&q);
while (q--) {
build();
ans1=inf;ans2=-inf;tot=0;
dp(1);
printf("%lld %lld %lld\n",tot,ans1,ans2);
for (int i=1;i<=K;i++) b[a[i]]=0;
}
return 0;
}
【bzoj3611】 大工程的更多相关文章
- [Bzoj3611]大工程(虚树+DP)
Description 题目链接 Solution 在虚树上跑DP即可 关于虚树的建立,是维护一个最右链的过程 关键代码如下: sort(A+1,A+k+1,cmp);//按dfs序排序 s[top= ...
- [HEOI2014][bzoj3611] 大工程 [虚树+dp]
题面: 传送门 思路: 又是一道虚树入门级的题目,但是这道题的实际难点在于dp 首先,这道题是可以点分治做的,而且因为6s时限随便浪,所以写点分治也不是不可以 但是,dp因为$O\left(n\rig ...
- [BZOJ3611][Heoi2014]大工程
[BZOJ3611][Heoi2014]大工程 试题描述 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上. 在 ...
- 【BZOJ3611】大工程(虚树,动态规划)
[BZOJ3611]大工程(虚树,动态规划) 题面 BZOJ Description 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树 ...
- 【BZOJ3611】[Heoi2014]大工程 欧拉序+ST表+单调栈
[BZOJ3611][Heoi2014]大工程 Description 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶 ...
- [BZOJ3611][Heoi2014]大工程(虚树上DP)
3611: [Heoi2014]大工程 Time Limit: 60 Sec Memory Limit: 512 MBSubmit: 2464 Solved: 1104[Submit][Statu ...
- BZOJ3611:[HEOI2014]大工程(树形DP,虚树)
Description 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上. 在 2 个国家 a,b 之间建一条新通 ...
- BZOJ2286 [Sdoi2011]消耗战 和 BZOJ3611 [Heoi2014]大工程
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6371 Solved: 2496[Submit][Statu ...
- 刷题总结——大工程(bzoj3611)
题目: Description 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上. 在 2 个国家 a,b 之间建 ...
- BZOJ3611 [Heoi2014]大工程 【虚树】
题目 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上. 在 2 个国家 a,b 之间建一条新通道需要的代价为树上 a ...
随机推荐
- js,onblur后下一个控件获取焦点判断、html当前活跃控件、jquery版本查看、jquery查看浏览器版本、setTimeout&setInterval
需求: input控件在失去焦点后直接做验证,验证通不过的话,显示相应错误.但是如果失去焦点后点击的下个控件是比较特殊的控件(比如,退出系统),那么不执行验证操作,直接退出系统(防止在系统退出前,还显 ...
- [转]考虑 PHP 5.0~5.6 各版本兼容性的 cURL 文件上传
FROM : https://segmentfault.com/a/1190000000725185 最近做的一个需求,要通过PHP调用cURL,以multipart/form-data格式上传文件. ...
- NOI2018准备 Day8
清北学堂入学测试,6道题凑了363分,平均466才能达到省选班的程度,差距不小. 今天突然感觉最大的BOSS是搜索,虽然每次都写崩...... 3个小时写了一道DP没写出来 但我不会忘记,我的首个目标 ...
- Ros集成开发环境配置
参考资料: http://blog.csdn.net/yangziluomu/article/details/50848357 ROS使用IDE Eclipse http://blog.csdn.ne ...
- JAVA CDI 学习(1) - @Inject基本用法
CDI(Contexts and Dependency Injection 上下文依赖注入),是JAVA官方提供的依赖注入实现,可用于Dynamic Web Module中,先给3篇老外的文章,写得很 ...
- Logging的这点小事
真正做项目,才发觉Logging的好处和学问.自己胡搞的时候,常常使用System.out.println作为输出. 但实际的项目,尤其是library比较多的时候,好好配置好Logging,才能在后 ...
- java中String、StringBuffer、StringBuilder的区别
java中String.StringBuffer.StringBuilder是编程中经常使用的字符串类,他们之间的区别也是经常在面试中会问到的问题.现在总结一下,看看他们的不同与相同. 1.可变与不可 ...
- 谈对象 MVC 和 多端
什么是对象? 我是单身狗,我没有对象:我是C程序猿,我没有对象:我是程序猿,我只会new一个对象. 言归正传,想想从一个电商网站上买一个东西,“进入首页,搜索商品,选型购买,登录下单,支付完成”,这里 ...
- 别再迷信 zepto 了
希望网上公开课的老师们不要再讲移动端网页用zepto了,坑了无数鸟啊 ~~~. 1.自己/公司/项目组所写和所积累(网上下的)的js函数都是以jQuery插件的写法来写的,如果要换到zepto上的话那 ...
- HttpClient4.5 SSL访问工具类
要从网上找一个HttpClient SSL访问工具类太难了,原因是HttpClient版本太多了,稍有差别就不能用,最后笔者干脆自己封装了一个访问HTTPS并绕过证书工具类. 主要是基于新版本Http ...