Codeforces 744A. Hongcow Builds A Nation
A. Hongcow Builds A Nation
题意: 现在有 n 个点 ,m 条边组成了一个无向图 , 其中有 k 个特殊点, 这些特殊点之间不能连通 ,问可以再多加几条边?
因为$x^2+y^2<=(x+y)^2$,所以找出所有连通块,枚举一个特殊连通块,与其他非特殊连通块合在一起。一个连通块贡献答案:$\frac{Size^2-Size}{2}-已有边$
#include< cstdio >
typedef long long ll;
template
inline void read(T&x)
{
x=0;bool f=0;char c=getchar();
while((c<'0'||c>'9')&&c!='-') c=getchar();if(c=='-')f=1, c=getchar();
while(c>='0'&&c<='9'){x=x*10+c-'0'; c=getchar();}
x=f?-x:x;
}
const int MAXN(1010);
int n,m,k,a,b,f[MAXN],c[MAXN],size[MAXN],have[MAXN],que[MAXN],tl,Ans,tmp,now;
bool bf[MAXN];
int gf(int x){return x==f[x]?x:f[x]=gf(f[x]);}
int max(int a,int b){return a>b?a:b;}
void Union(int a,int b)
{
int fa=gf(a),fb=gf(b);
if(fa!=fb)
{
size[fa]+=size[fb];
have[fa]+=have[fb];
f[fb]=fa;
}
have[fa]++;
}
int Edge(int x)
{
return (size[x]*(size[x]-1))/2-have[x];
}
int main()
{
// freopen("C.in","r",stdin);
// freopen("C.out","w",stdout);
read(n);read(m);read(k);
for(int i=1;i<=n;i++)f[i]=i,size[i]=1;
for(int i=1;i<=k;i++)read(c[i]);
for(int i=1;i<=m;i++)
{
read(a);read(b); Union(a,b);
}
for(int i=1;i<=n;i++)
if(!bf[gf(i)])que[++tl]=gf(i),bf[gf(i)]=1;
for(int i=1;i<=k;i++)
{
bf[gf(c[i])]=0;
Ans+=Edge(gf(c[i]));
}
// fprintf(stderr,"%d\n",tl);
for(int i=1;i<=tl;i++)
if(bf[que[i]])
{
size[0]+=size[que[i]];
have[0]+=have[que[i]];
}
// fprintf(stderr,"Ans%d\n",Ans);
// fprintf(stderr,"0:%d\n",Edge(0));
now=Ans;
for(int i=1;i<=k;i++)
{
tmp=now;
size[0]+=size[gf(c[i])];
have[0]+=have[gf(c[i])];
tmp-=Edge(gf(c[i]));
tmp+=Edge(0);
// fprintf(stderr,"s0:%d\n",size[0]);
// fprintf(stderr,"h0:%d\n",have[0]);
// fprintf(stderr,"e0:%d\n",Edge(0));
// fprintf(stderr,"tmp:%d\n",tmp);
size[0]-=size[gf(c[i])];
have[0]-=have[gf(c[i])];
Ans=max(Ans,tmp);
}
printf("%d\n",Ans);
return 0;
}

浙公网安备 33010602011771号