3611: [Heoi2014]大工程

Time Limit: 60 Sec  Memory Limit: 512 MB
Submit: 408  Solved: 190
[Submit][Status][Discuss]

Description

国家有一个大工程,要给一个非常大的交通网络里建一些新的通道。 
我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上。 
在 2 个国家 a,b 之间建一条新通道需要的代价为树上 a,b 的最短路径。
 现在国家有很多个计划,每个计划都是这样,我们选中了 k 个点,然后在它们两两之间 新建 C(k,2)条 新通道。
现在对于每个计划,我们想知道:
 1.这些新通道的代价和
 2.这些新通道中代价最小的是多少 
3.这些新通道中代价最大的是多少

Input

第一行 n 表示点数。

 接下来 n-1 行,每行两个数 a,b 表示 a 和 b 之间有一条边。
点从 1 开始标号。 接下来一行 q 表示计划数。
对每个计划有 2 行,第一行 k 表示这个计划选中了几个点。
 第二行用空格隔开的 k 个互不相同的数表示选了哪 k 个点。

Output

输出 q 行,每行三个数分别表示代价和,最小代价,最大代价。

 

Sample Input

10
2 1
3 2
4 1
5 2
6 4
7 5
8 6
9 7
10 9
5
2
5 4
2
10 4
2
5 2
2
6 1
2
6 1

Sample Output

3 3 3
6 6 6
1 1 1
2 2 2
2 2 2

HINT

n<=1000000

q<=50000并且保证所有k之和<=2*n 

Source

【思路】

虚树+树上DP

对每次询问构造虚树,在虚树上进行DP。

ans1和ans2即树上的最长/短链问题,利用前缀和思想可以求解。

   设sum[x] = ∑(sum[y] + w * size[y]); 则有

    ans += ∑((sum[y] + w * size[y]) * (size[x] - size[y]));

  其中size[x]表示以x为根的子树包含的询问点数目,w为x->y的边长。

   好大的工程=-=

【代码】

 #include<cstdio>
#include<vector>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std; const int N = +;
const int INF = 1e9+1e9;
const int D = ; typedef long long LL;
vector<int> G[N],g[N];
int d[N],dfn[N];
int n,q,dfsc; void adde(int u,int v) {
if(u!=v) G[u].push_back(v); else return;
}
bool cmp(const int& lhs,const int& rhs) {
return dfn[lhs]<dfn[rhs];
}
////////////////////////////////////////////////////lca which cuts down about 5000ms
int siz[N],top[N],son[N],fa[N];
void dfs1(int u) {
siz[u]=,son[]=; dfn[u]=++dfsc;
for(int i=;i<g[u].size();i++) {
int v=g[u][i];
if(v!=fa[u]) {
fa[v]=u; d[v]=d[u]+;
dfs1(v);
siz[u]+=siz[v];
if(siz[v]>siz[son[u]]) son[u]=v;
}
}
}
void dfs2(int u,int tp) {
top[u]=tp;
if(son[u]) dfs2(son[u],tp);
for(int i=;i<g[u].size();i++) {
int v=g[u][i];
if(v!=son[u] && v!=fa[u]) dfs2(v,v);
}
}
int LCA(int u,int v) {
while(top[u]!=top[v])
if(d[top[u]]>=d[top[v]]) u=fa[top[u]];
else v=fa[top[v]];
return d[u]<d[v]? u:v;
}
////////////////////////////////////////////////
LL sum[N],ans;
int ans1,ans2,mi[N],mx[N],sz[N]; bool ifq[N];
int dp(int u) {
sum[u]=; sz[u]=ifq[u];
mi[u]=ifq[u]? :INF;
mx[u]=ifq[u]? :-INF;
for(int i=;i<G[u].size();i++) {
int v=G[u][i],w=d[v]-d[u];
dp(v);
sz[u]+=sz[v];
sum[u]+=sum[v]+sz[v]*w;
ans1=min(ans1,mi[u]+mi[v]+w);
ans2=max(ans2,mx[u]+mx[v]+w);
mi[u]=min(mi[u],mi[v]+w);
mx[u]=max(mx[u],mx[v]+w);
}
for(int i=;i<G[u].size();i++) {
int v=G[u][i],w=d[v]-d[u];
ans+=(sum[v]+sz[v]*w)*(sz[u]-sz[v]);
}
ifq[u]=; G[u].clear(); //clear
}
void read(int& x) {
char c=getchar();while(!isdigit(c))c=getchar();
x=;while(isdigit(c))x=x*+c-'' , c=getchar();
}
void solve() {
int tot=,top=,k;
static int st[N],h[N];
read(k);
for(int i=;i<=k;i++) read(h[i]),ifq[h[i]]=;
sort(h+,h+k+,cmp);
st[++top]=;
for(int i=;i<=k;i++) {
int p=h[i],lca=LCA(p,st[top]);
for(;;) {
if(d[lca]>=d[st[top-]]) {
adde(lca,st[top]);
top--;
if(st[top]!=lca) st[++top]=lca;
break;
}
adde(st[top-],st[top]); top--;
}
if(st[top]!=p) st[++top]=p;
}
while(--top) adde(st[top],st[top+]);
ans=ans2= , ans1=INF;
dp();
printf("%lld %d %d\n",ans,ans1,ans2);
}
int main() {
read(n);
int u,v;
for(int i=;i<n;i++) {
read(u),read(v);
g[u].push_back(v) , g[v].push_back(u);
}
d[]=; dfs1(); dfs2(,);
read(q);
while(q--) solve();
return ;
}

bzoj 3611 [Heoi2014]大工程(虚树+DP)的更多相关文章

  1. bzoj 3611[Heoi2014]大工程 虚树+dp

    题意: 给一棵树 每次选 k 个关键点,然后在它们两两之间 新建 C(k,2)条 新通道. 求: 1.这些新通道的代价和 2.这些新通道中代价最小的是多少 3.这些新通道中代价最大的是多少 分析:较常 ...

  2. BZOJ.3611.[HEOI2014]大工程(虚树 树形DP)

    题目链接 要求的和.最大值.最小值好像都可以通过O(n)的树形DP做,总询问点数<=2n. 于是建虚树就可以了.具体DP见DP()函数,维护三个值sum[],mx[],mn[]. sum[]要开 ...

  3. bzoj 3611: [Heoi2014]大工程 虚树

    题目: 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道. 我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上. 在 2 个国家 a,b 之间建一条新通道需要的代价为树上 ...

  4. BZOJ 3611 [Heoi2014]大工程 ——虚树

    虚树第二题.... 同BZOJ2286 #include <map> #include <cmath> #include <queue> #include < ...

  5. bzoj 3611(洛谷 4103) [Heoi2014]大工程——虚树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3611 https://www.luogu.org/problemnew/show/P4103 ...

  6. luogu P4103 [HEOI2014]大工程 虚树 + 树形 DP

    Description 国家有一个大工程,要给一个非常大的交通网络里建一些新的通道.  我们这个国家位置非常特殊,可以看成是一个单位边权的树,城市位于顶点上.  在 2 个国家 a,b 之间建一条新通 ...

  7. 洛谷P4103 [HEOI2014]大工程(虚树 树形dp)

    题意 链接 Sol 虚树. 首先建出虚树,然后直接树形dp就行了. 最大最小值直接维护子树内到该节点的最大值,然后合并两棵子树的时候更新一下答案. 任意两点的路径和可以考虑每条边两边的贡献,\(d[x ...

  8. [HEOI2014][bzoj3611] 大工程 [虚树+dp]

    题面: 传送门 思路: 又是一道虚树入门级的题目,但是这道题的实际难点在于dp 首先,这道题是可以点分治做的,而且因为6s时限随便浪,所以写点分治也不是不可以 但是,dp因为$O\left(n\rig ...

  9. bzoj 3611: [Heoi2014]大工程 && bzoj 2286: [Sdoi2011消耗战

    放波建虚树的模板. 大概是用一个栈维护根节点到当前关键点的一条链,把其他深度大于lca的都弹出去. 每次做完记得复原. 还有sort的时候一定要加cmp!!! bzoj 3611 #include&l ...

随机推荐

  1. delphi 功能函数大全-备份用

    function CheckTask(ExeFileName: string): Boolean;constPROCESS_TERMINATE=$0001;varContinueLoop: BOOL; ...

  2. Excel数据导入到oracle

    打开pl/sql,如图所示界面,点击菜单栏中的T00LS     ODBC Imtorper,打开ODBC Importer选项框       在Data fromODBC页中选择需要导入的文件的格式 ...

  3. 数据库(学习整理)----4--Oracle数据查询(基础点1)

    其他: 计算机中的内存是线性的,一维. length('')计算字符的个数,而不是字节的个数 Oracle中的日期类型和数值类型的数据可以做运算符(>,=,<,<>)比较 如果 ...

  4. ZOJ 1076 Gene Assembly(LIS+路径打印 贪心)

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=76 题目大意:在一个DNA上,给定许多基因的起始位置和结束位置,求出这 ...

  5. dp题目

    从别的地方看来,最近一直在啃DP,有个目标,更有动力了. 1.Robberies 连接 :http://acm.hdu.edu.cn/showproblem.php?pid=2955      背包; ...

  6. 【HDU1402】【FFT】A * B Problem Plus

    Problem Description Calculate A * B. Input Each line will contain two integers A and B. Process to e ...

  7. Git (2)

    要使用Git首先遇到的问题是怎么把文件加到库中. 很简单. 新建一个目录,然后git init. 完成上述工作之后的唯一改动是在当前目录下生成了一个.git的子目录.这个子目录是一个集中的数据库,包含 ...

  8. 【随记】还原SQL Server数据库步骤

    情景:在一台机器上备份数据库,然后在另一台机器上还原数据库,可能会出现错误提示:System.Data.SqlClient.SqlError: 备份集中的数据库备份与现有的 'XXX' 数据库不同. ...

  9. string与char* 互相转换以及周边问题

    先插一个小知识点 string str = "abc" str += 'd'; cout<<str<<endl;  //"abcd"   ...

  10. MySQL驱动阅读------Connection连接的建立,基于JDBC-----5.1.26

    一般获取数据库连接的程序 Class.forName("com.mysql.jdbc.Driver"); final Connection connection = (Connec ...