zoukankan      html  css  js  c++  java
  • DP————最小覆盖问题

    原题:https://www.luogu.org/problem/P2279

    题解转载自:https://www.luogu.org/blog/contributation/solution-p2279

    思路

    找最低没被覆盖到的点,并在它的祖父处设一个消防站。考虑到这个点的所有子孙后代都已经被覆盖了,因此这时覆盖祖父能盖到更多额外的点,并保证结果不会更差。

    很多思路是用dfs或堆求取最低节点,实际上没必要,只要预处理出深度(边输入边处理)并排序,碰到已覆盖就跳过,未覆盖就在祖父处设消防站,ans++。

    问题在于怎样才能判断这个点覆盖到了没有。对于儿子或孙子覆盖他,可以在在儿子处设站时就标记它;而对于父亲和祖父覆盖他,可以用儿子对父亲的映射f来解决;问题在于兄弟。其实,可以用o数组维护“离i最近的消防站到i的距离”,当o[父亲]==1时,就能确定它是否被覆盖。

    代码

    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #define N 2020
    #define FOR(i,a,b) for(int i=a;i<=b;i++)
    using namespace std;
    int n,b[N],f[N],d[N],o[N],ans,u,v,w;
    bool cmp(int x,int y){return d[x]>d[y];}
    int main(){
        scanf("%d",&n);b[1]=1,o[1]=o[0]=N;
        FOR(i,2,n) scanf("%d",&f[i]),d[i]=d[f[i]]+1,b[i]=i,o[i]=N;
        sort(b+1,b+n+1,cmp);
        FOR(i,1,n){
            v=b[i],w=f[v],u=f[f[v]];
            o[v]=min(o[v],min(o[w]+1,o[u]+2));
            if(o[v]>2){
                o[u]=0,ans++;
                o[f[u]]=min(o[f[u]],1),o[f[f[u]]]=min(o[f[f[u]]],2);
            }
        }printf("%d",ans);
    }
    

     这种方法的普适性很强,可以解决半径为k的最小覆盖问题。而且不用存图。只需要把维护“父亲和爷爷”改成维护“上位k位祖先”即可,复杂度O(N*K),常数也很小。

  • 相关阅读:
    Java中的集合类
    Java中的包装类
    Java中的多线程总结(转)
    Java开发中的23种设计模式详解 (转)
    Java中异常处理和设计
    Jmeter的ForEach控制器
    Jmeter事务控制器
    Jmeter下线程顺序启动
    Jmeter跨线程调用参数
    Jmeter使用Python
  • 原文地址:https://www.cnblogs.com/myhnb/p/11324137.html
Copyright © 2011-2022 走看看