zoukankan      html  css  js  c++  java
  • BZOJ4557:[JLOI2016/SHOI2016]侦察守卫——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4557

    小R和B神正在玩一款游戏。这款游戏的地图由N个点和N-1条无向边组成,每条无向边连接两个点,且地图是连通的。换句话说,游戏的地图是一棵有N个节点的树。

    游戏中有一种道具叫做侦查守卫,当一名玩家在一个点上放置侦查守卫后,它可以监视这个点以及与这个点的距离在D以内的所有点。这里两个点之间的距离定义为它们在树上的距离,也就是两个点之间唯一的简单路径上所经过边的条数。在一个点上放置侦查守卫需要付出一定的代价,在不同点放置守卫的代价可能不同。

    现在小R知道了所有B神可能会出现的位置,请你计算监视所有这些位置的最小代价。

    dp状态很好想,但是这个式子我菜我是真的推不出来,其他的巨佬切题的速度叹为观止,只能感叹我的智商摆在这里。

    参考:https://www.luogu.org/blog/zcysky/solution-p3267

    一眼是个树形dp,二眼$d$很小,可以直接做成一维状态,那么直接设$f[i][j]$为$i$子树从$i$往下$j$层都没有覆盖的代价,$g[i][j]$为$i$的子树全覆盖,往上还可以覆盖$j$层的代价。二者正好是互补的。

    (PS:层数也包括i本身,换句话说,$j=0$时$i$并没有被覆盖,我在这里纠结了很久。)

    (PPS:既然$g[i][j]$都可以覆盖上$j$层,那它也能覆盖下$j$层。)

    之后对于dp式子慢慢剖析因为我自己都云里雾里的

    边界就是当点$u$为关键点时$f[u][0]=g[u][0]=w[u]$这个点一定是要放一个的,如果不是的话显然我们就不需要放了,初值为0。

    初始化就不说了。

    对于每个儿子结点v,我们有:

    $g[u][j]=min(g[u][j]+f[v][j],g[v][j+1]+f[u][j+1])$(所以明白f和g是互补的才能看懂)

    当然也有可能出现这种的:$g[u][j]=min(g[u][j],g[u][j+1])$

    推完g来推f,首先$f[u][0]=g[u][0]$因为此时二者状态等价。

    然后显然的,$f[u][j]+=f[v][j-1]$

    以及也有可能出现这种的:$f[u][j]=min(f[u][j],f[u][j-1])$

    (所以其实核心还是在状态含义上而非式子,含义搞懂式子就很显然了。)

    #include<map>
    #include<cmath>
    #include<stack>
    #include<queue>
    #include<cstdio>
    #include<cctype>
    #include<vector>
    #include<cstdlib>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    using namespace std;
    typedef long long ll;
    const int N=5e5+5;
    const int D=21;
    const int INF=1e9;
    inline int read(){
        int X=0,w=0;char ch=0;
        while(!isdigit(ch)){w|=ch=='-';ch=getchar();}
        while(isdigit(ch))X=(X<<3)+(X<<1)+(ch^48),ch=getchar();
        return w?-X:X;
    }
    struct node{
        int to,nxt;
    }e[N*2];
    int n,d,m,cnt,head[N],w[N];
    int f[N][D],g[N][D];
    bool im[N];
    inline void add(int u,int v){
        e[++cnt].to=v;e[cnt].nxt=head[u];head[u]=cnt;
    }
    void dfs(int u,int fa){
        if(im[u])f[u][0]=g[u][0]=w[u];
        for(int i=1;i<=d;i++)g[u][i]=w[u];
        g[u][d+1]=INF;
        for(int i=head[u];i;i=e[i].nxt){
            int v=e[i].to;
            if(v==fa)continue;
            dfs(v,u);
            for(int j=d;j>=0;j--)g[u][j]=min(g[u][j]+f[v][j],g[v][j+1]+f[u][j+1]);
            for(int j=d;j>=0;j--)g[u][j]=min(g[u][j],g[u][j+1]);
            f[u][0]=g[u][0];
            for(int j=1;j<=d+1;j++)f[u][j]+=f[v][j-1];
            for(int j=1;j<=d+1;j++)f[u][j]=min(f[u][j],f[u][j-1]);
        }
    }
    int main(){
        n=read(),d=read();
        for(int i=1;i<=n;i++)w[i]=read();
        m=read();
        for(int i=1;i<=m;i++)im[read()]=1;
        for(int i=1;i<n;i++){
            int u=read(),v=read();
            add(u,v);add(v,u);
        }
        dfs(1,0);
        printf("%d
    ",f[1][0]);
        return 0;
    }

    +++++++++++++++++++++++++++++++++++++++++++

    +本文作者:luyouqi233。               +

    +欢迎访问我的博客:http://www.cnblogs.com/luyouqi233/+

    +++++++++++++++++++++++++++++++++++++++++++

  • 相关阅读:
    java7底层源码
    google的collection
    2017年八大顶尖的技术趋势
    【译】STM32L4x6系列用户手册第四章
    FRDM-KL43开发板驱动段式液晶SLCD的实现方法
    如何根据丝印查找相关的产品型号
    Arduino Tian开发板:一个功能强大的天气预报中心
    在STM32F746G-DISCO开发板上使用Nabto + FreeRTOS的演示热泵应用
    为LPC1549 LPCXpresso评估板开发基于mbed的项目
    使用LPCXpresso开发板调试外部的电路板
  • 原文地址:https://www.cnblogs.com/luyouqi233/p/9193761.html
Copyright © 2011-2022 走看看