BZOJ 2286: [Sdoi2011]消耗战
2286: [Sdoi2011消耗战
Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 2082  Solved: 736
[Submit][Status][Discuss]
Description
Input
第一行一个整数n,代表岛屿数量。
接下来n-1行,每行三个整数u,v,w,代表u号岛屿和v号岛屿由一条代价为c的桥梁直接相连,保证1<=u,v<=n且1<=c<=100000。
第n+1行,一个整数m,代表敌方机器能使用的次数。
接下来m行,每行一个整数ki,代表第i次后,有ki个岛屿资源丰富,接下来k个整数h1,h2,…hk,表示资源丰富岛屿的编号。
Output
输出有m行,分别代表每次任务的最小代价。
Sample Input
1 5 13
1 9 6
2 1 19
2 4 8
2 3 91
5 6 8
7 5 4
7 8 31
10 7 9
3
2 10 6
4 5 7 8 3
3 9 4 6
Sample Output
32
22
HINT
对于100%的数据,2<=n<=250000,m>=1,sigma(ki)<=500000,1<=ki<=n-1
Source
虚树入门题,如果你想知道虚树如何构造:http://user.qzone.qq.com/872191552/main
DP
记dis[i]为点i到根节点的路径中最小的边的权值
f[i]=min(dis[i],sigma f[son[i]]);
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
#include<stack>
#define maxn 500100
#define llg long long
using namespace std;
struct node
{
llg df,po;
}d[maxn];
llg i,j,k,n,m,p[maxn][],dad[maxn],bj[maxn],dfs[maxn],cnt,T,deep[maxn],dis[maxn],se[maxn],q,ss[maxn],open,dl[maxn];
vector <llg> a[maxn],c[maxn],val[maxn];
stack <llg> z; inline int getint(){
char c=getchar(); llg w=,q=;
while(c!='-' && ( c<'' || c>'')) c=getchar();
if(c=='-') c=getchar(),q=;
while(c>='' && c<='') w=w*+c-'',c=getchar();
return q?-w:w;
} void link(llg x,llg y,llg z) {a[x].push_back(y); val[x].push_back(z); val[y].push_back(z); a[y].push_back(x);} void link_(llg x,llg y) {c[x].push_back(y); c[y].push_back(x);} bool cmp(const node&a,const node&b){return a.df<b.df;} void find_dad_deep(llg x,llg di)
{
bj[x]=; dfs[x]=++cnt; dis[x]=di;
llg w=a[x].size();
for (llg i=;i<w;i++)
if (!bj[a[x][i]])
{
dad[a[x][i]]=x; p[a[x][i]][]=x; deep[a[x][i]]=deep[x]+;
find_dad_deep(a[x][i],min(di,val[x][i]));
}
} void build_lca()
{
for (i=;(<<i)<=n;i++)
for (j=;j<=n;j++)
p[j][i]=p[p[j][i-]][i-];
} llg lca(llg x,llg y)
{
if (deep[x]<deep[y]) swap(x,y);
for (llg i=;i>=;i--) if (deep[p[x][i]]>=deep[y]) x=p[x][i];
if (x==y) return x;
for (llg i=;i>=;i--)
if (p[x][i]!=p[y][i]) {x=p[x][i]; y=p[y][i];}
return dad[x];
} llg st[maxn],top; void make_tree(llg x)
{
llg L=,g1,g2;
while(top)
{
g1=st[top];g2=st[top-];L=lca(g1,x);
if(deep[g2]>deep[L]) link_(g2,g1),top--;
else if(deep[g2]<deep[L]) {link_(g1,L);top--;break;}
else break;
}
if(st[top]!=L) st[++top]=L;
st[++top]=x;
} void ceshi()
{
for (llg i=;i<=n;i++)
{
llg w=c[i].size();
for (llg j=;j<w;j++) if (i<c[i][j]) cout<<i<<" "<<c[i][j]<<endl;
}
} llg dp(llg x)
{
bj[x]=;
dl[++open]=x;
llg w=c[x].size(),tot=;
for (llg i=;i<w;i++)
if (!bj[c[x][i]])
tot+=dp(c[x][i]);
if (ss[x]) return dis[x];
return min(tot,dis[x]);
} int main()
{
freopen("a.in","r",stdin); freopen("a.out","w",stdout);
cin>>n;
for (i=;i<n;i++)
{
llg x,y,z;
x=getint(),y=getint(),z=getint();
link(x,y,z);
}
deep[]=;
llg ggg=(llg)1e60;
find_dad_deep(,ggg); //dis[1]=0;
build_lca();
cin>>T; for (i=;i<=n;i++) bj[i]=;
while (T--)
{
m=getint();
llg x;
for (i=;i<=open;i++) {ss[se[i]]=; c[dl[i]].clear(); bj[dl[i]]=;}
q=,open=;
c[].clear(); bj[]=; ss[]=;
for (i=;i<=m;i++) {x=getint(); d[i].po=x; d[i].df=dfs[x]; q++; ss[x]=; se[q]=x; }
sort(d+,d+m+,cmp);
top=;
if (d[].po!=) st[++top]=;
for (i=;i<=m;i++) make_tree(d[i].po);
while(top!=) link_(st[top],st[top-]),top--;
//if (q<=0) printf("0\n"); else
printf("%lld\n",dp());
//ceshi();
//cout<<endl;
}
return ;
}
BZOJ 2286: [Sdoi2011]消耗战的更多相关文章
- bzoj 2286: [Sdoi2011]消耗战 虚树+树dp
		
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MB[Submit][Status][Discuss] Description 在一 ...
 - bzoj 2286 [Sdoi2011]消耗战(虚树+树上DP)
		
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2286 [题意] 给定一棵树,切断一条树边代价为ci,有m个询问,每次问使得1号点与查询 ...
 - bzoj 3611: [Heoi2014]大工程  && bzoj 2286: [Sdoi2011消耗战
		
放波建虚树的模板. 大概是用一个栈维护根节点到当前关键点的一条链,把其他深度大于lca的都弹出去. 每次做完记得复原. 还有sort的时候一定要加cmp!!! bzoj 3611 #include&l ...
 - BZOJ 2286: [Sdoi2011]消耗战 虚树 树形dp 动态规划 dfs序
		
https://www.lydsy.com/JudgeOnline/problem.php?id=2286 wa了两次因为lca犯了zz错误 这道题如果不多次询问的话就是裸dp. 一棵树上多次询问,且 ...
 - BZOJ.2286.[SDOI2011]消耗战(虚树 树形DP)
		
题目链接 BZOJ 洛谷P2495 树形DP,对于每棵子树要么逐个删除其中要删除的边,要么直接断连向父节点的边. 如果当前点需要删除,那么直接断不需要再管子树. 复杂度O(m*n). 对于两个要删除的 ...
 - BZOJ 2286 [Sdoi2011]消耗战(虚树+树形DP)
		
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2286 [题目大意] 出一棵边权树,每次给出一些关键点,求最小边割集, 使得1点与各个关 ...
 - bzoj 2286: [Sdoi2011消耗战
		
#include<cstdio> #include<iostream> #define M 1000009 #define N 250009 #define ll long l ...
 - bzoj 2286 [Sdoi2011]消耗战   虚树+dp
		
题目大意:多次给出关键点,求切断边使所有关键点与1断开的最小费用 分析:每次造出虚树,dp[i]表示将i和i子树与父亲断开费用 对于父亲x,儿子y ①y为关键点:\(dp[x]\)+=\(dismn( ...
 - BZOJ 2286: [Sdoi2011消耗战 [DP 虚树]
		
传送门 题意: 删除价值和最小的边使得$1$号点与$k$个关键点不连通 一个树形DP...但是询问多次,保证总的关键点数为$O(n)$ 先说一下这个$DP$ $f[i]$表示子树$i$中的关键点与$1 ...
 
随机推荐
- LaTex 基础
			
一.文档 \documentclass{article} %book, report, letter 二.宏包 \usepackage{amsmath} 三.正文 \begin{document} \ ...
 - logback使用总结
			
filter: http://aub.iteye.com/blog/1110008 http://aub.iteye.com/blog/1101222 Logback Log4j的创始人Ceki Gü ...
 - php mysql_num_rows() 与 mysql_affected_rows()
			
mysql_num_rows(data) 函数返回结果集中行的数目. data 结果集.该结果集从 mysql_query() 的调用中得到. 此命令仅对 SELECT 语句有效.要取得被 INSER ...
 - [分享]IOS开发-简单实现搜索框显示历史记录的本地缓存及搜索历史每次只能获取到一个的解决方案
			
注:原文:http://www.zhimengzhe.com/IOSkaifa/40433.html 1.首先,我们需要对进行过搜索的textField的输入内容进行一个NSUserDefaults的 ...
 - 生成器(generator)内部解析
			
#http://kb.cnblogs.com/page/87128/(未看完)
 - Linux下安装Redis3.2.4
			
安装: 通过wget方式直接在linux上下载Redis $ wget http://download.redis.io/releases/redis-3.2.4.tar.gz , 默认下载到路径是r ...
 - iOS下Html页面中input获取焦点弹出键盘时挡住input解决方案—scrollIntoView()
			
问题描述 iOS系统下,移动web页面,inpu获取焦点弹出系统虚拟键盘时,偶尔会出现挡住input的情况,尽管概率不大,但是十分影响用户体验. 问题重现 原始页面:页面中有header.main.f ...
 - AD6电气规则错误报告中英文对照
			
Ⅰ:Error Reporting 错误报告 A:Violations Associated with Buses 有关总线电气错误的各类型(共 12 项) ◆ bus indices out of ...
 - 深入Python(4):深拷贝和浅拷贝
			
一.前奏:熟悉Python内存管理 在Python中,变量在第一次赋值时自动声明,在创建---也就是赋值的时候,解释器会根据语法和右侧的操作数来决定新对象的类型. 引用计数器:一个内部跟踪变量 引用计 ...
 - AngularJS Best Practices: SEO
			
Google can execute AJAX & JavaScript for indexing, you can read the below link for more detailed ...