bzoj 2286 [Sdoi2011]消耗战 虚树+dp
题目大意:多次给出关键点,求切断边使所有关键点与1断开的最小费用
分析:每次造出虚树,dp[i]表示将i和i子树与父亲断开费用
对于父亲x,儿子y
①y为关键点:\(dp[x]\)+=\(dismn(x,y)\)
②y不为关键点:要么断y,要么断y所有子树
\(dp[x]\)+=\(min(dismn(x,y),dp[y])\)
=========================================================
关于兼容性的一种讨论
dismn(x,y)直接改为dismn(1,x)预处理算可以吗
当然不行?
交一发,A
兼容性:若要算到y,则y到1路径中没有关键点
①情况无影响
②情况中若子树中最小值算到了上面,断y定更优,而断y又变成父节点的①②情况讨论
所以是可以的咯
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <cctype>
using namespace std;
typedef long long LL;
const int M=250007;
inline int rd(){
int x=0;bool f=1;char c=getchar();
for(;!isdigit(c);c=getchar())if(c=='-')f=0;
for(;isdigit(c);c=getchar())x=x*10+c-48;
return f?x:-x;
}
int n,m;
LL dp[M];
LL dis[M];
LL w[M][20];
int ln[M];
int dep[M],sz[M],son[M],pre[M];
int top[M],dfn[M],pid[M],T;
int que[M],kd[M];
bool cmp(int x,int y){return dfn[x]<dfn[y];}
int st[M],tot;
int g[M],te;
struct edge{int y,next;LL d;}e[M<<1];
void addedge(int x,int y,LL d){
e[++te].y=y;e[te].d=d;e[te].next=g[x];g[x]=te;
}
int hd[M],td;
struct link{int y,next;}dw[M];
void addlink(int x,int y){
if(x==y)return;
dw[++td].y=y;dw[td].next=hd[x];hd[x]=td;
}
void dfs1(int x){
sz[x]=1;
int p,y;
for(p=g[x];p;p=e[p].next)
if((y=e[p].y)!=pre[x]){
dep[y]=dep[x]+1;
dis[y]=e[p].d;
pre[y]=x;
dfs1(y);
sz[x]+=sz[y];
if(sz[y]>sz[son[x]]) son[x]=y;
}
}
void dfs2(int x){
pid[dfn[x]=++T]=x;
if(son[x]){
top[son[x]]=top[x];
dfs2(son[x]);
}
int p,y;
for(p=g[x];p;p=e[p].next)
if((y=e[p].y)!=pre[x]&&y!=son[x]){
top[y]=y;
dfs2(y);
}
}
int LCA(int x,int y){
while(top[x]!=top[y]){
if(dep[top[x]]<dep[top[y]]) swap(x,y);
x=pre[top[x]];
}
if(dep[x]>dep[y]) swap(x,y);
return x;
}
void vbuild(int z){
int i,x,anc;
sort(que+1,que+z+1,cmp);
for(i=1;i<z;i++){
anc=LCA(que[i],que[i+1]);
hd[anc]=0; kd[anc]=2; dp[anc]=0;
}
hd[1]=0; dp[1]=0;//*****
for(i=1;i<=z;i++){
x=que[i];
hd[x]=0; kd[x]=1; dp[x]=0;
}
td=0;
tot=0;
st[++tot]=1;
for(i=1;i<=z;i++){
x=que[i];
anc=LCA(x,st[tot]);
if(anc==st[tot]){
st[++tot]=x;
continue;
}
else{
while( tot>1 && dep[anc]<=dep[st[tot-1]]){
addlink(st[tot-1],st[tot]);
tot--;
}
addlink(anc,st[tot]);
st[tot]=anc;
st[++tot]=x;
}
}
for(i=1;i<tot;i++) addlink(st[i],st[i+1]);
}
LL getm(int x,int y){
int l=ln[y-x+1];
return min(w[x][l],w[y-(1<<l)+1][l]);
}
void init(){
int i,j,l;
for(i=2;i<=n;i++) ln[i]=ln[i>>1]+1;
for(i=1;i<=n;i++) w[i][0]=dis[pid[i]];
for(i=n;i>0;i--){
l=ln[n-i+1];
for(j=1;j<=l;j++) w[i][j]=min(w[i][j-1],w[i+(1<<j-1)][j-1]);
}
}
LL getw(int x,int y){
LL res=1LL<<61;
while(dep[top[x]]>dep[y]){
res=min(res,getm(dfn[top[x]],dfn[x]));
x=pre[top[x]];
}
if(x!=y) res=min(res,getm(dfn[y]+1,dfn[x]));
return res;
}
void dfs(int x){
int p,y;
LL tp;
for(p=hd[x];p;p=dw[p].next){
y=dw[p].y;
tp=getw(y,x);
if(kd[y]==1) dp[x]+=tp;
else{
dfs(y);
dp[x]+=min(tp,dp[y]);
}
}
}
int main(){
int i,x,y,z;
n=rd();
for(i=1;i<n;i++){
x=rd(),y=rd(),z=rd();
addedge(x,y,z);
addedge(y,x,z);
}
dep[1]=pre[1]=0;
dis[1]=0;
dfs1(1);
top[1]=1;
dfs2(1);
init();
m=rd();
while(m--){
z=rd();
for(i=1;i<=z;i++) que[i]=rd();
vbuild(z);
dfs(1);
printf("%lld\n",dp[1]);
}
return 0;
}
bzoj 2286 [Sdoi2011]消耗战 虚树+dp的更多相关文章
- bzoj 2286: [Sdoi2011]消耗战 虚树+树dp
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MB[Submit][Status][Discuss] Description 在一 ...
- 【BZOJ】2286: [Sdoi2011]消耗战 虚树+DP
[题意]给定n个点的带边权树,每次询问给定ki个特殊点,求隔离点1和特殊点的最小代价.n<=250000,Σki<=500000. [算法]虚树+DP [题解]考虑普通树上的dp,设f[x ...
- 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 #include <map> #include <ctime> #include <cmath&g ...
- BZOJ 2286: [Sdoi2011]消耗战 虚树
Description 在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总部在编号为1的岛屿,而且他们已经没有足够多的能源维系战斗,我军 ...
- [BZOJ2286][SDOI2011]消耗战(虚树DP)
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 4998 Solved: 1867[Submit][Statu ...
- BZOJ 2286: [Sdoi2011]消耗战
2286: [Sdoi2011消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2082 Solved: 736[Submit][Status] ...
- BZOJ 3572 [HNOI2014]世界树 (虚树+DP)
题面:BZOJ传送门 洛谷传送门 题目大意:略 细节贼多的虚树$DP$ 先考虑只有一次询问的情况 一个节点$x$可能被它子树内的一个到x距离最小的特殊点管辖,还可能被管辖fa[x]的特殊点管辖 跑两次 ...
随机推荐
- string转化为int方法
int intA = 0; 1.intA =int.Parse(str); 2.int.TryParse(str, out intA); 3.intA = Convert.ToInt32(str);以 ...
- 我也谈“the difference between Factory, Service, and Provider in Angular”
看完这篇文章之后的理解与实践:原文地址:http://tylermcginnis.com/angularjs-factory-vs-service-vs-provider/ <!doctype ...
- QWidget QMainWindow QDialog 三个基类的区别
Qt类是一个提供所需的像全局变量一样的大量不同的标识符的命名空间.通常情况下,你可以忽略这个类.QObject和一些其它类继承了它,所以在这个Qt命名空间中定义的所有标识符通常情况下都可以无限制的使用 ...
- 转:Web 测试的创作与调试技术
摘要:学习有关 Visual Studio 2005 Web 测试的更多知识,包括 Web 测试引擎和记录器如何工作,以及如何创建有效的 Web 测试. 本页内容 读者 简介 记录一个 Web 测试 ...
- Python3基础 使用list() 生成一个空列表
镇场诗: 诚听如来语,顿舍世间名与利.愿做地藏徒,广演是经阎浮提. 愿尽吾所学,成就一良心博客.愿诸后来人,重现智慧清净体.-------------------------------------- ...
- 微信小程序登陆流程
#1:session_key和openId是什么?session_key 官方说明为:session_key是微信服务器生成的针对用户数据进行加密签名的密钥session_key的用途(1)对wx.g ...
- StrictMode使用详解
http://hb.qq.com/a/20110914/000054.htm http://www.android100.org/html/201204/25/1097.html http://www ...
- UML类图基本画法
转自:http://www.cnblogs.com/import-this/p/5330501.html 1. 类 类包括属性方法 下图的生命就是属性 代谢就是方法 2.基本关系 a.泛化(Gener ...
- Django之路: 模版篇
一.Django 模版 上章是介绍简单的把django.http.HttpResponse的内容显示到网页上,下面就介绍以下如何使用渲染模版的方法来显示内容.本节代码是基于Django 1.8,但 ...
- 汇总博客常见的api接口地址(windows live write)
汇总博客常见的api接口地址(windows live write) 1. cnblogs 日志地址,直接输入 http://www.cnblogs.com/xxxxx/ api接口 http://w ...