还是虚树的题目啊...

如果只有一个询问,我们这么考虑,可以设dp[x]为只删除x子树内和x到父亲的边,使得x这棵子树内的能源岛屿都与x的父亲不连通的最小花费。

这样如果x本身是能源岛屿,那么dp[x]=fe[x],否则dp[x]=min(fe[x],sum{dp[son[x]]})类似这样。(fe表示父亲边,fe[1]=inf)

那么有了多组询问我们就在虚树上搞这个,需要注意的是虚树上的一条父子边就对应着原树上一条父子链,而权值就是那个父子链的最小值。可以用倍增一起维护掉。

需要注意的是...要用long long啊

#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <string.h>
#include <math.h>
#include <set>
#include <map>
using namespace std;
typedef long long ll;
int inf=1000000000;
ll inf_ll=10000000000000000LL;
#define gc getchar()
int g_i()
{
int tmp=0; bool fu=0; char s;
while(s=gc,s!='-'&&(s<'0'||s>'9')) ;
if(s=='-') fu=1; else tmp=s-'0';
while(s=gc,s>='0'&&s<='9') tmp=tmp*10+s-'0';
if(fu) return -tmp; else return tmp;
}
#define gi g_i()
#define pob
#define pc(x) putchar(x)
namespace ib {char b[100];}
inline void pll(ll x)
{
if(x==0) {pc(48); return;}
if(x<0) {pc('-'); x=-x;}
char *s=ib::b;
while(x) *(++s)=x%10, x/=10;
while(s!=ib::b) pc((*(s--))+48);
}
#define SZ 1234567
#define D 20
#define _els ;else
//real tree
namespace rt
{
int n,fst[SZ],nxt[SZ],vb[SZ],vc[SZ],fa[SZ],dep[SZ],M=0,dfsn[SZ],C=0,sz[SZ],fe[SZ];
int up[SZ][D],mine[SZ][D];
void ad_de(int a,int b,int c) {++M; nxt[M]=fst[a]; fst[a]=M; vb[M]=b; vc[M]=c;}
void adde(int a,int b,int c) {ad_de(a,b,c); ad_de(b,a,c);}
void dfs(int p)
{
dfsn[p]=++C; sz[p]=1;
for(int e=fst[p];e;e=nxt[e])
{
int b=vb[e]; if(b==fa[p]) continue;
fa[b]=up[b][0]=p; fe[b]=vc[e]; mine[b][0]=fe[b];
dep[b]=dep[p]+1; dfs(b); sz[p]+=sz[b];
}
}
void build()
{
dfs(1);
for(int g=1;g<D;g++)
{
for(int i=1;i<=n;i++) mine[i][g]=inf;
}
for(int g=1;g<D;g++)
{
for(int i=1;i<=n;i++)
{
if(!up[i][g-1]) continue;
up[i][g]=up[up[i][g-1]][g-1];
mine[i][g]=min(mine[i][g-1],mine[up[i][g-1]][g-1]);
}
}
}
//jump up (x=fa[x]) until dep[x]=d
int jmp(int x,int d)
{
for(int i=D-1;i>=0;i--)
{
if(!up[x][i]||dep[up[x][i]]<d)_els x=up[x][i];
}
return x;
}
int gmin(int x,int d)
{
int minn=inf;
for(int i=D-1;i>=0;i--)
{
if(!up[x][i]||dep[up[x][i]]<d)_els minn=min(minn,mine[x][i]), x=up[x][i];
}
return minn;
}
int lca(int x,int y)
{
if(dep[x]>dep[y]) swap(x,y);
y=jmp(y,dep[x]);
if(x==y) return x;
for(int i=D-1;i>=0;i--)
{
if(up[x][i]!=up[y][i]) x=up[x][i], y=up[y][i];
}
return fa[x];
}
}
//virtual tree
namespace vt
{
#define f_ first
#define s_ second
typedef pair<int,int> pii;
//vs: points in vtree
int sn,ss[SZ],vn,vs[SZ],stn=0,st[SZ],vfa[SZ],fc[SZ],nc[SZ];
ll f[SZ],vfe[SZ];
bool saf[SZ];
bool cmp_dfsn(int a,int b) {return rt::dfsn[a]<rt::dfsn[b];}
ll dp(int x)
{
if(!saf[x]) return f[x]=vfe[x];
ll sum=0;
for(int c=fc[x];c;c=nc[c]) sum+=dp(c);
return f[x]=min((ll)vfe[x],sum);
}
void build()
{
vn=stn=0;
ss[++sn]=1;
sort(ss+1,ss+1+sn,cmp_dfsn);
for(int i=1;i<=sn;i++) vs[++vn]=ss[i], saf[ss[i]]=i==1;
for(int i=1;i<=sn;i++)
{
int x=ss[i];
if(!stn) {st[++stn]=x; vfa[x]=0; continue;}
int lca=rt::lca(x,st[stn]);
for(;rt::dep[st[stn]]>rt::dep[lca];--stn)
{
if(rt::dep[st[stn-1]]<=rt::dep[lca]) vfa[st[stn]]=lca;
}
if(st[stn]!=lca)
{
vs[++vn]=lca;
saf[lca]=1;
vfa[lca]=st[stn];
st[++stn]=lca;
}
vfa[x]=lca; st[++stn]=x;
}
sort(vs+1,vs+1+vn,cmp_dfsn);
for(int i=1;i<=vn;i++) fc[vs[i]]=0;
for(int i=1;i<=vn;i++)
{
int x=vs[i]; f[x]=inf_ll;
if(i>1) vfe[x]=rt::gmin(x,rt::dep[vfa[x]]);
else vfe[x]=inf_ll;
int f=vfa[x];
nc[x]=fc[f]; fc[f]=x;
}
pll(dp(1)); pc(10);
}
}
int main()
{
rt::n=gi;
for(int i=1;i<rt::n;i++)
{
int x=gi,y=gi,z=gi;
rt::adde(x,y,z);
}
rt::build();
int q=gi;
while(q--)
{
vt::sn=gi;
for(int i=1;i<=vt::sn;i++) vt::ss[i]=gi;
vt::build();
}
}

bzoj2286 消耗战的更多相关文章

  1. [Bzoj2286]消耗战(虚树+DP)

    Description 题目链接 Solution 在虚树上跑DP即可 Code #include <cstdio> #include <algorithm> #include ...

  2. [SDOI2011][bzoj2286] 消耗战 [虚树+dp]

    题面: 传送门 思路: 看到所有询问中的点数总和是十万级别的,就想到用虚树~\(≧▽≦)/~啦 首先,树形dp应该是很明显可以看出来的: 设dp[u]表示以u为根的子树(不包括u)中的宝藏岛全部切断的 ...

  3. [bzoj2286][Sdoi 2011]消耗战

    [bzoj2286]消耗战 标签: 虚树 DP 题目链接 题解 很容易找出\(O(mn)\)的做法. 只需要每次都dp一遍. 但是m和n是同阶的,所以这样肯定会T的. 注意到dp的时候有很多节点是不需 ...

  4. DP——由蒟蒻到神犇的进阶之路

    开始更新咯 DP专题[题目来源BZOJ] 一.树形DP 1.bzoj2286消耗战 题解:因为是树形结构,一个点与根节点不联通,删一条边即可, 于是我们就可以简化这棵树,把有用的信息建立一颗虚树,然后 ...

  5. 【BZOJ2286】消耗战(虚树,动态规划)

    [BZOJ2286]消耗战(虚树,动态规划) 题面 BZOJ Description 在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总 ...

  6. [BZOJ2286][SDOI2011]消耗战(虚树DP)

    2286: [Sdoi2011]消耗战 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 4998  Solved: 1867[Submit][Statu ...

  7. 【BZOJ2286】[Sdoi2011]消耗战 虚树

    [BZOJ2286][Sdoi2011]消耗战 Description 在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总部在编号为1的 ...

  8. 虚树+【BZOJ2286】【SDOI2011】消耗战(虚树)(DP)

    先看一道题: [BZOJ2286][SDOI2011]消耗战 Description 在一场战争中,战场由n个岛屿和n−1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总 ...

  9. [BZOJ2286][Sdoi2011]消耗战(虚树上DP)

    2286: [Sdoi2011]消耗战 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 6457  Solved: 2533[Submit][Statu ...

随机推荐

  1. phonegap创建的ios项目推送消息出现闪退现象

    使用phonegap创建的ios项目,推送消息时,当程序在前台运行或者在后台运行状态下,推送消息过来,可以解析并且跳转: 但是在程序从后台退出的状态下,当消息推送过来的时候,点击通知栏,打开程序,程序 ...

  2. Android studio修改Logcat颜色

    Android studio默认的Logcat配色不利于阅读,我们可以修改自定义自己的颜色配置

  3. UITextFiled,UITextView长度限制

    长度限制用到的地方很多,但是需求都不一样.有的要求全部字符按一个处理,有的要求英文字母按一个,中文按两个,emoji按四个.这样就会遇到各种各样奇怪的问题,再被虐了无数次后,终于解决掉了.下面就来写写 ...

  4. APNS远程推送(转发)

    /*****************************************2************************************************/ /****** ...

  5. iOS-H5学习篇-02

    H5-自学笔记-2016年09月06日 一:各种标签的练习 Html和CSS的关系 学习web前端开发基础技术需要掌握:HTML.CSS.JavaScript语言. 1.1.1. HTML是网页内容的 ...

  6. Hadoop概念学习系列之Hadoop 生态系统(十二)

    当下 Hadoop 已经成长为一个庞大的生态体系,只要和海量数据相关的领域,都有 Hadoop 的身影.下图是一个 Hadoop 生态系统的图谱,详细列举了在 Hadoop 这个生态系统中出现的各种数 ...

  7. (视频) 《快速创建网站》3.4 网站改版3分钟搞定 - WordPress主题安装和备份

    本文是<快速创建网站>系列的第8篇,如果你还没有看过之前的内容,建议你点击以下目录中的章节先阅读其他内容再回到本文. 访问本系列目录,请点击:http://devopshub.cn/tag ...

  8. Activiti之 Exclusive Gateway

    一.Exclusive Gateway Exclusive Gateway(也称为XOR网关或更多技术基于数据的排他网关)经常用做决定流程的流转方向.当流程到达该网关的时候,所有的流出序列流到按照已定 ...

  9. mysql学习笔记(一)

    my建表操作 创建表 create Table <表名> ( 字段名1,数据类型 [列级约束] [默认值], 字段名2,数据类型 [列级约束] [默认值], ... [表级约束], [co ...

  10. JVM探索之——内存管理(二)

    上篇文章我们介绍了JVM所管理的内存结构也就是运行时数据区(Run-Time Data Areas),现在我们将介绍JVM的内存分配与回收静态内存分配与动态内存分配 JVM的内存分配主要分为两种:静态 ...