题目链接:点击打开链接
#1127 : 二分图三·二分图最小点覆盖和最大独立集
时间限制:
10000ms
单点时限:
1000ms
内存限制:
256MB
-
5 4 3 2 1 3 5 4 1 5
样例输出
-
2 3
描述
在上次安排完相亲之后又过了挺长时间,大家好像都差不多见过面了。不过相亲这个事不是说那么容易的,所以Nettle的姑姑打算收集一下之前的情况并再安排一次相亲。所以现在摆在Nettle面前的有2个问题:
1.姑姑想要了解之前所有相亲的情况。对于任一个一次相亲,只要跟参与相亲的两人交流就可以得到这次相亲的情况。如果一个人参加了多次相亲,那么跟他交流就可以知道这几次相亲的情况。那么问题来了,挖掘技术到底哪家强姑姑最少需要跟多少人进行交流可以了解到所有相亲的情况。
2.因为春节快要结束了,姑姑打算给这些人再安排一次集体相亲。集体相亲也就是所有人在一起相亲,不再安排一对一对的进行相亲。但是姑姑有个条件,要求所有参与相亲的人之前都没有见过。也就是说在之前的每一次相亲中的两人不会被同时邀请来参加这次集体相亲。那么问题又来了,姑姑最多可以让多少人参与这个集体相亲。
输入
第1行:2个正整数,N,M(N表示点数 2≤N≤1,000,M表示边数1≤M≤5,000)
第2..M+1行:每行两个整数u,v,表示一条无向边(u,v)
输出
第1行:1个整数,表示最小点覆盖数
第2行:1个整数,表示最大独立集数
#include <cstdio>
#include <iostream>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <string.h>
#include <string>
#define MEM(a,x) memset(a,x,sizeof a)
#define eps 1e-8
#define MOD 10009
#define MAXN 10010
#define INF 99999999
#define ll __int64
#define bug cout<<"here"<<endl
#define fread freopen("ceshi.txt","r",stdin)
#define fwrite freopen("out.txt","w",stdout)
using namespace std;
void read(int &x)
{
char ch;
x=0;
while(ch=getchar(),ch!=' '&&ch!='\n')
{
x=x*10+ch-'0';
}
}
int n,m;
int mp[1050][1050];
int link[1050];
int vis[1050];
int ans;
bool dfs(int u)
{
for(int i=1;i<=n;i++)
{
if(mp[u][i]&&!vis[i])
{
vis[i]=1;
if(link[i]==-1||dfs(link[i]))
{
link[i]=u;
return 1;
}
}
}
return 0;
}
void hungary()
{
MEM(link,-1);
for(int i=1;i<=n;i++)
{
MEM(vis,0);
if(dfs(i))
ans++;
}
}
int main()
{
// fread;
while(scanf("%d%d",&n,&m)!=EOF)
{
MEM(mp,0);
while(m--)
{
int u,v;
scanf("%d%d",&u,&v);
// u--; v--;
mp[u][v]=mp[v][u]=1;//无向图建图
}
ans=0;
hungary();
printf("%d\n",ans/2);//答案除以2 最小点覆盖数=二分图匹配数
printf("%d\n",n-ans/2);//最大独立集数=点的个数-二分图匹配数
}
return 0;
}