【jzoj4896】【兔子】【二分答案】

题目大意

在一片草原上有N个兔子窝,每个窝里住着一只兔子,有M条路径连接这些窝。更特殊地是,至多只有一个兔子窝有3条或更多的路径与它相连,其它的兔子窝只有1条或2条路径与其相连。换句话讲,这些兔子窝之前的路径构成一张N个点、M条边的无向连通图,而度数大于2的点至多有1个。兔子们决定把其中K个兔子窝扩建成临时避难所。当危险来临时,每只兔子均会同时前往距离它最近的避难所躲避,路程中花费的时间在数值上等于经过的路径条数。为了在最短的时间内让所有兔子脱离危险,请你安排一种建造避难所的方式,使最后一只到达避难所的兔子所花费的时间尽量少。

解题思路

首先我们可以二分答案,枚举一个可以覆盖到关键点的点,这样可以删掉这个点覆盖的点,这样就不可能出现环只有链,可以直接算出需要放多少个点。

code

#include
#include
#include
#define LL long long
#define max(x,y) ((x>y)?x:y)
#define fo(i,j,k) for(int i=j;i<=k;i++)
#define fd(i,j,k) for(int i=j;i>=k;i--)
using namespace std;
int const maxn=1000,maxm=1500,inf=1e9;
int n,m,k,gra,size,f[maxn+10],u[maxn+10],du[maxn+10],begin[maxn+10],
to[maxm*2+10],next[maxm*2+10],vis[maxn+10][10];
void insert(int u,int v){
    to[++gra]=v;
    next[gra]=begin[u];
    begin[u]=gra;
}
void dfs(int now,int pre,int dis,int op){
    vis[now][op]=1;size++;
    if(!dis)return;
    for(int i=begin[now];i;i=next[i])
        if((to[i]!=pre)&&(!vis[to[i]][op])&&((op!=2)||(!vis[to[i]][1])))dfs(to[i],now,dis-1,op);
}
int main(){
    //freopen("rabbit.in","r",stdin);
    //freopen("rabbit.out","w",stdout);
    freopen("d.in","r",stdin);
    freopen("d.out","w",stdout);
    scanf("%d%d%d",&n,&m,&k);
    fo(i,1,m){
        int u,v;scanf("%d%d",&u,&v);
        du[u]++;du[v]++;insert(u,v);insert(v,u);
    }
    int root=1;
    fo(i,1,n)if(du[i]>2){root=i;break;}
    int l=0,r=n,ans,put,tmp;
    for(;l!=r;){
        ans=(l+r)/2;
        fo(i,1,n)vis[i][0]=0;
        dfs(root,0,ans,0);
        put=inf;
        fo(i,1,n)
            if(vis[i][0]){
                fo(j,1,n)vis[j][1]=vis[j][2]=0;
                dfs(i,0,ans,1);tmp=1;
                fo(j,1,n)
                    if((!vis[j][1])&&(!vis[j][2])){
                        size=0;dfs(j,0,inf,2);
                        tmp+=(size-1)/(ans*2+1)+1;
                    }
                put=min(put,tmp);
            }
        if(put<=k)r=ans;
        else l=ans+1;
    }
    printf("%d",l);
    return 0;
}

你可能感兴趣的:(jzoj,二分,三分,贪心)