题目大意
在一片草原上有N个兔子窝,每个窝里住着一只兔子,有M条路径连接这些窝。更特殊地是,至多只有一个兔子窝有3条或更多的路径与它相连,其它的兔子窝只有1条或2条路径与其相连。换句话讲,这些兔子窝之前的路径构成一张N个点、M条边的无向连通图,而度数大于2的点至多有1个。兔子们决定把其中K个兔子窝扩建成临时避难所。当危险来临时,每只兔子均会同时前往距离它最近的避难所躲避,路程中花费的时间在数值上等于经过的路径条数。为了在最短的时间内让所有兔子脱离危险,请你安排一种建造避难所的方式,使最后一只到达避难所的兔子所花费的时间尽量少。
解题思路
首先我们可以二分答案,枚举一个可以覆盖到关键点的点,这样可以删掉这个点覆盖的点,这样就不可能出现环只有链,可以直接算出需要放多少个点。
code
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;
}