[BZOJ2286][Sdoi2011]消耗战(虚树上DP)
2286: [Sdoi2011]消耗战
Time Limit: 20 Sec Memory Limit: 512 MB
Submit: 6457 Solved: 2533
[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
Solution
一道虚树上dp的板子题。
设d[x]为根到x路径上的最小边,那么对于标记点,显然只能决策d[x],而非标记点可以决策min(d[x],sigma(d[y])),y为x的所有儿子。
有两种建虚树的方法,第一种是网上的常规建法,第二种是从yyb哪儿学到的,要更简单一些。两种我都写了的。
注意下第一种方法的insert函数里的注释,因为我们这样特殊处理了所以第一种方法不用打标记,dp时也不用限制决策;
而第二种方法是把整棵虚树都建出来了的,因此必须打标记,而且要随时注意回溯时清空标记(不能直接memset就不用我说了吧)。
Code 1
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=,INF=0x3f3f3f3f;
inline int read(){
int x=,w=;char ch=;
while(!isdigit(ch)) w|=ch=='-',ch=getchar();
while(isdigit(ch)) x=(x<<)+(x<<)+(ch^),ch=getchar();
return w?-x:x;
}
struct edge{
int v,w,last;
}e[N<<];
int tot,tail[N];
inline void add(int x,int y,int z){
e[++tot]=(edge){y,z,tail[x]};
tail[x]=tot;
}
int idx,dfn[N],dep[N],d[N],f[N][];
int dfs(int x,int pre){
dfn[x]=++idx;dep[x]=dep[pre]+;f[x][]=pre;
for(int i=;;++i)
if(f[x][i-]) f[x][i]=f[f[x][i-]][i-];
else break;
for(int p=tail[x];p;p=e[p].last){
int &v=e[p].v,&w=e[p].w;
if(v==pre) continue;
d[v]=min(d[x],w);
dfs(v,x);
}
}
int LCA(int x,int y){
if(dep[x]<dep[y]) x^=y^=x^=y;
for(int i=;i>=;--i) if(dep[f[x][i]]>=dep[y]) x=f[x][i];
if(x==y) return x;
for(int i=;i>=;--i) if(f[x][i]^f[y][i]) x=f[x][i],y=f[y][i];
return f[x][];
}
int n,m,k,tp,st[N],a[N];
void insert(int x){
if(tp==){st[++tp]=x;return ;}
int lca=LCA(x,st[tp]);
if(lca==st[tp]){
/*任意时刻如果lca=st[tp]的话那么lca一定是标记点(lca=1除外),既然lca和x都是标记点,
那么x就不用管了呗,相当于把x的决策合并到lca上去了。*/
return ;
}
while(tp>&&dfn[st[tp-]]>=dfn[lca]) add(st[tp-],st[tp],),tp--;
if(st[tp]^lca) add(lca,st[tp],),st[tp]=lca;
st[++tp]=x;
}
LL dp(int x){
if(!tail[x]) return d[x];
LL sum=;
for(int p=tail[x];p;p=e[p].last)
sum+=dp(e[p].v);
tail[x]=;
if(x==) return sum;
else return min(sum,1ll*d[x]);
}
bool cmp(int x,int y){return dfn[x]<dfn[y];}
int main(){
n=read();
for(int i=,x,y,z;i<n;++i){
x=read(),y=read(),z=read();
add(x,y,z),add(y,x,z);
}
d[]=INF;
dfs(,);
m=read();
tot=;memset(tail,,sizeof tail);
st[tp=]=;
while(m--){
tot=;
k=read();
for(int i=;i<=k;++i) a[i]=read();
sort(a+,a+k+,cmp);
for(int i=;i<=k;++i) insert(a[i]);
while(tp>) add(st[tp-],st[tp],),--tp;
printf("%lld\n",dp());
}
return ;
}
BZOJ2286
Code 2
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=,INF=0x3f3f3f3f;
inline int read(){
int x=,w=;char ch=;
while(!isdigit(ch)) w|=ch=='-',ch=getchar();
while(isdigit(ch)) x=(x<<)+(x<<)+(ch^),ch=getchar();
return w?-x:x;
}
struct edge{
int v,w,last;
}e[N<<];
int tot,tail[N];
inline void add(int x,int y,int z){
e[++tot]=(edge){y,z,tail[x]};
tail[x]=tot;
}
int idx,dfn[N],dep[N],d[N],low[N],f[N][];
int dfs(int x,int pre){
dfn[x]=++idx;dep[x]=dep[pre]+;f[x][]=pre;
for(int i=;;++i)
if(f[x][i-]) f[x][i]=f[f[x][i-]][i-];
else break;
for(int p=tail[x];p;p=e[p].last){
int &v=e[p].v,&w=e[p].w;
if(v==pre) continue;
d[v]=min(d[x],w);
dfs(v,x);
}
low[x]=idx;
}
int LCA(int x,int y){
if(dep[x]<dep[y]) x^=y^=x^=y;
for(int i=;i>=;--i) if(dep[f[x][i]]>=dep[y]) x=f[x][i];
if(x==y) return x;
for(int i=;i>=;--i) if(f[x][i]^f[y][i]) x=f[x][i],y=f[y][i];
return f[x][];
}
bool tag[N];
int n,m,k,st[N<<],a[N<<];
LL dp(int x){
if(!tail[x]) {tag[x]=;return d[x];}
LL sum=;
for(int p=tail[x];p;p=e[p].last)
sum+=dp(e[p].v);
tail[x]=;
if(x==) return sum;
if(tag[x]) {tag[x]=;return d[x];}
return min(sum,1ll*d[x]);
}
bool cmp(int x,int y){return dfn[x]<dfn[y];}
int main(){
n=read();
for(int i=,x,y,z;i<n;++i){
x=read(),y=read(),z=read();
add(x,y,z),add(y,x,z);
}
d[]=INF;
dfs(,);
m=read();
tot=;memset(tail,,sizeof tail);
while(m--){
tot=;
k=read();
for(int i=;i<=k;++i) a[i]=read(),tag[a[i]]=true;
a[++k]=; //手动添加1号节点
sort(a+,a+k+,cmp);
for(int i=k;i>;--i)
a[++k]=LCA(a[i],a[i-]);
sort(a+,a+k+,cmp);
k=unique(a+,a+k+)-a-;
for(int i=,tp=;i<=k;++i){
while(tp&&low[st[tp]]<dfn[a[i]]) --tp;
if(tp) add(st[tp],a[i],);
st[++tp]=a[i];
}
printf("%lld\n",dp());
}
return ;
}
BZOJ2286
[BZOJ2286][Sdoi2011]消耗战(虚树上DP)的更多相关文章
- [BZOJ2286][SDOI2011]消耗战(虚树DP)
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 4998 Solved: 1867[Submit][Statu ...
- BZOJ2286: [Sdoi2011]消耗战(虚树/树形DP)
Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 5246 Solved: 1978[Submit][Status][Discuss] Descript ...
- bzoj2286: [Sdoi2011]消耗战 虚树
在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总部在编号为1的岛屿,而且他们已经没有足够多的能源维系战斗,我军胜利在望.已知在其他k个 ...
- 【BZOJ】2286: [Sdoi2011]消耗战 虚树+DP
[题意]给定n个点的带边权树,每次询问给定ki个特殊点,求隔离点1和特殊点的最小代价.n<=250000,Σki<=500000. [算法]虚树+DP [题解]考虑普通树上的dp,设f[x ...
- bzoj 2286 [Sdoi2011]消耗战 虚树+dp
题目大意:多次给出关键点,求切断边使所有关键点与1断开的最小费用 分析:每次造出虚树,dp[i]表示将i和i子树与父亲断开费用 对于父亲x,儿子y ①y为关键点:\(dp[x]\)+=\(dismn( ...
- BZOJ 2286 [Sdoi2011]消耗战 ——虚树
虚树第一题. 大概就是建一颗只与询问有关的更小的新树,然后在虚树上DP #include <map> #include <ctime> #include <cmath&g ...
- [BZOJ3611][Heoi2014]大工程(虚树上DP)
3611: [Heoi2014]大工程 Time Limit: 60 Sec Memory Limit: 512 MBSubmit: 2464 Solved: 1104[Submit][Statu ...
- BZOJ2286 [Sdoi2011]消耗战 【虚树 + 树形Dp】
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MB Submit: 4261 Solved: 1552 [Submit][Sta ...
- 【BZOJ-2286】消耗战 虚树 + 树形DP
2286: [Sdoi2011消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2120 Solved: 752[Submit][Status] ...
随机推荐
- 使用git保存管理代码
1.git是个代码版本管理软件,类似SVN github是个网站,提供git服务,我们只需要注册个账号,就可以使用它的git服务,不需要自己部署git系统 git需要先在电脑端安装,安装完成后,讲产生 ...
- JVM内存模型入门
JVM内存模型入门 本文是学习笔记,原文地址在:https://www.bilibili.com/video/av62009886 综述 其实没有太多新东西 JVM主要分为五个区域:栈区.堆区.本地方 ...
- uoj #46[清华集训2014]玄学
uoj 因为询问是关于一段连续区间内的操作的,所以对操作构建线段树,这里每个点维护若干个不交的区间,每个区间\((l,r,a,b)\)表示区间\([l,r]\)内的数要变成\(ax+b\) 每次把新操 ...
- 在不同电脑设备之间, 同步 VSCode 的插件和配置
前提有一个码云或者github的账户,以下是我用github的示例(只有第二步不一样): Step1. 安装 同步插件"Settings Sync" Step2. 进入github ...
- 利用python3 调用zabbix接口完成批量加聚合图形(screens)
在上一篇博客中,我们完成的利用python3 调用zabbix接口批量增加主机,增加主机的item,增加主机的图形! 接下来我们完成批量增加主机的screen 首先我们要增加screen需要哪些参数呢 ...
- python: 基本数据类型 与 内置函数 知识整理
列表 list.append(val) #末尾追加,直接改变无返回 list.inert(2,val) #插入到指定位置 list.extend(mylist1) #list会被改变 list2=li ...
- NDK: ant 错误 [javah] Exception in thread "main" java.lang.NullPointerException 多种解决办法
1.错误提示内容 2.ant脚本对应的内容 <?xml version="1.0" encoding="UTF-8"?> <!-- ===== ...
- MySQL面试题(二)
● 请你说一说mysql的四种隔离状态 参考回答: Mysql主要包含四种隔离状态: 事务隔离级别 脏读 不可重复读 幻读 读未提交(read-uncommitted) 是 是 是 不可重复读(rea ...
- HQL实现模糊查询
hibernate 实现模糊查询两种传参方式,其实各个方法的实质都是一样的,只不过传递参数的方法稍微有点区别 public List<User> getUsers(String id){ ...
- 求二叉搜索树的第k小的节点
题目描述: /** * 给定一棵二叉搜索树,请找出其中的第k小的结点. * 例如, (5,3,7,2,4,6,8)中, * 按结点数值大小顺序第三小结点的值为4. * 这是层序遍历: * 5 * 3 ...