题目大意:多次给出关键点,求切断边使所有关键点与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的更多相关文章

  1. bzoj 2286: [Sdoi2011]消耗战 虚树+树dp

    2286: [Sdoi2011]消耗战 Time Limit: 20 Sec  Memory Limit: 512 MB[Submit][Status][Discuss] Description 在一 ...

  2. 【BZOJ】2286: [Sdoi2011]消耗战 虚树+DP

    [题意]给定n个点的带边权树,每次询问给定ki个特殊点,求隔离点1和特殊点的最小代价.n<=250000,Σki<=500000. [算法]虚树+DP [题解]考虑普通树上的dp,设f[x ...

  3. BZOJ 2286: [Sdoi2011]消耗战 虚树 树形dp 动态规划 dfs序

    https://www.lydsy.com/JudgeOnline/problem.php?id=2286 wa了两次因为lca犯了zz错误 这道题如果不多次询问的话就是裸dp. 一棵树上多次询问,且 ...

  4. BZOJ.2286.[SDOI2011]消耗战(虚树 树形DP)

    题目链接 BZOJ 洛谷P2495 树形DP,对于每棵子树要么逐个删除其中要删除的边,要么直接断连向父节点的边. 如果当前点需要删除,那么直接断不需要再管子树. 复杂度O(m*n). 对于两个要删除的 ...

  5. BZOJ 2286 [Sdoi2011]消耗战 ——虚树

    虚树第一题. 大概就是建一颗只与询问有关的更小的新树,然后在虚树上DP #include <map> #include <ctime> #include <cmath&g ...

  6. BZOJ 2286: [Sdoi2011]消耗战 虚树

    Description 在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总部在编号为1的岛屿,而且他们已经没有足够多的能源维系战斗,我军 ...

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

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

  8. BZOJ 2286: [Sdoi2011]消耗战

    2286: [Sdoi2011消耗战 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2082  Solved: 736[Submit][Status] ...

  9. BZOJ 3572 [HNOI2014]世界树 (虚树+DP)

    题面:BZOJ传送门 洛谷传送门 题目大意:略 细节贼多的虚树$DP$ 先考虑只有一次询问的情况 一个节点$x$可能被它子树内的一个到x距离最小的特殊点管辖,还可能被管辖fa[x]的特殊点管辖 跑两次 ...

随机推荐

  1. Entity Framework 学习初级篇7--基本操作:增加、更新、删除、事务

    本节,直接写通过代码来学习.这些基本操作都比较简单,与这些基本操作相关的内容在之前的1至6节基本介绍完毕. l           增加: 方法1:使用AddToXXX(xxx)方法:实例代码如下: ...

  2. android点滴之HandlerThread的用法

    转载请注明出处:http://blog.csdn.net/lskshz/article/details/25364909 一.介绍 HandlerThread继承自Thread,当线程开启时,也就是它 ...

  3. jna

    http://topmanopensource.iteye.com/blog/1752355

  4. 在ubuntu 上创建 ssl 证书

    soap webservice 调试工具: soap UI, 可以下载下来玩一玩. Introduction TLS, or transport layer security, and its pre ...

  5. GuideActivity.java引导界面:

    这是谷歌官方给我们提供的一个兼容低版本安卓设备的软件包,里面包囊了只有在安卓3.0以上可以使用的api. 而viewpager就是其中之一利用它,我们可以做很多事情,从最简单的导航,到页面菜单等等.那 ...

  6. CentOS6.5 常用命令

    卸载 rpm -e XXX yum -y remove XXX 查看内核版本 cat /proc/version

  7. 8数码,欺我太甚!<bfs+康拓展开>

    不多述,直接上代码,至于康拓展开,以前的文章里有 #include<iostream> #include<cstdio> #include<queue> using ...

  8. composer 的快速安装

    Packagist 镜像 请各位使用本镜像的同学注意: 本镜像已经依照 composer 官方的数据源安全策略完全升级并支持 https 协议!请各位同学 按照下面所示的两个方法将 http://pa ...

  9. [iOS Animation]-CALayer 缓冲

    缓冲 生活和艺术一样,最美的永远是曲线. -- 爱德华布尔沃 - 利顿 在第九章“图层时间”中,我们讨论了动画时间和CAMediaTiming协议.现在我们来看一下另一个和时间相关的机制--所谓的缓冲 ...

  10. C语言实现求字符串子集问题

    这个问题在于实现Apriori算法的时候需要求各个频繁集的关联规则,而这时需要在求得最大的频繁集中求各个频繁集的真子集.然后在实现这一步的时候被卡主了,第一反应是用递归完成,但是面对数据挖掘中庞大的项 ...