【BZOJ2530】[Poi2011]Party (xia)构造

【BZOJ2530】[Poi2011]Party

Description

给定一张N(保证N是3的倍数)个节点M条边的图,并且保证该图存在一个大小至少为2N/3的团。
请输出该图的任意一个大小为N/3的团。 一个团的定义为节点的一个子集,该子集中的点两两有直接连边。 输入: 第一行是两个整数N,M。 接下来有M行,每行两个整数A,B,表示A和B有连边。保证无重边。 输出: N/3个整数,表示你找到的团。
数据范围: 

3<=N<=3000,[3/2 n(2/3 n -1)]/2<=M<=[n(n-1)/2]

Sample Input

6 10
2 5
1 4
1 5
2 4
1 3
4 5
4 6
3 5
3 4
3 6

Sample Output

2 4

题解:首先我们知道,在原图中找团,就是在补图中找出一些点使得两两之间不直接相连。

所以我的做法是:先搞出原图的补图,然后每次贪心选取(补图中)度数较少的点,将所有与它直接相连的点去掉,直到找出n/3个点。

结果一交上去就A了,看题解发现根本没必要贪心,随机选点就行了。

#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
int n,m;
int A[3010][3010],d[3010],vis[3010];
int rd()
{
	int ret=0,f=1;	char gc=getchar();
	while(gc<'0'||gc>'9')	{if(gc=='-')f=-f;	gc=getchar();}
	while(gc>='0'&&gc<='9')	ret=ret*10+gc-'0',gc=getchar();
	return ret*f;
}
int main()
{
	int i,j,a,b;
	n=rd(),m=rd();
	for(i=1;i<=n;i++)	for(d[i]=n-1,j=1;j<=i;j++)	A[i][j]=A[j][i]=1;
	for(i=1;i<=m;i++)	a=rd(),b=rd(),A[a][b]=A[b][a]=0,d[a]--,d[b]--;
	d[0]=n;
	for(i=1;i<=n/3;i++)
	{
		for(a=0,j=1;j<=n;j++)	if(!vis[j]&&d[j]<d[a])	a=j;
		vis[a]=1;
		if(i>1)	printf(" ");
		printf("%d",a);
		for(j=1;j<=n;j++)	if(A[a][j])	vis[j]=1;
	}
	return 0;
}
 
posted @ 2017-06-20 16:58  CQzhangyu  阅读(385)  评论(0编辑  收藏  举报