题目大意:多次给出关键点,求切断边使所有关键点与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. java网络之udp

    import java.net.*; /** * InetAddress用法 * 获取本地主机名,ip * 通过主机名,ip获取InetAddress对象 */ public class Demo{ ...

  2. public private proteccted区别

    public公共,加上这个修饰的类或属性,可以在同一个包或者别的包里面访问 private私有的,加上这个修饰的类或属性,只能在同类里访问,同包和别的包不能访问 protected保护,加上这个修饰的 ...

  3. SPFA中 正逆邻接表的建立

    正邻接表的建立: 先定义一个结构体: struct node { int r,v,w,next; }Map[]; 每输入一组数据 就通过Add函数加入到邻接表中,上图已经说得很明白了,结合着下面的代码 ...

  4. arm nop

    MOV R0,R0 这个语句相当于做一次无用功,也就相当于实现了NOP.       接下去就是怎么修改机器码的部分.先查询手册,查到MOV 的机器码是0xE1A0(此处可能不同,主要看自己IDA反汇 ...

  5. Java基于Socket文件传输示例

    http://www.blogjava.net/sterning/archive/2007/10/13/152508.html 最近需要进行网络传输大文件,于是对基于socket的文件传输作了一个初步 ...

  6. mariadb cache

    Since MariaDB Galera cluster versions 5.5.40 and 10.0.14 you can use the query cache. Earlier versio ...

  7. LB 高可扩展性集群(负载均衡集群)

    一.什么是负载均衡 首先我们先介绍一下什么是负载均衡: 负载平衡(Load balancing)是一种计算机网络技术,用来在多个计算机(计算机集群).网络连接.CPU.磁盘驱动器或其他资源中分配负载, ...

  8. (中等) CF 576D Flights for Regular Customers (#319 Div1 D题),矩阵快速幂。

    In the country there are exactly n cities numbered with positive integers from 1 to n. In each city ...

  9. Android Service生命周期 Service里面的onStartCommand()方法详解

    在Demo上,Start一个Service之后,执行顺序:onCreate - > onStartCommand 然后关闭应用,会重新执行上面两步. 但是把代码拷贝到游戏工程发现,关闭游戏后,只 ...

  10. php中print_r 和var_dump 打印变量的区别。

    <?php $arr = array(true); var_dump($arr); echo "<br/>"; print_r($arr); 结果如下: 说明 p ...