Description
Analysis
由于题目的特性,我们把极大团看成点,把点看成边。
然后会形成多个连通块。
对于每个连通块,若其中有
m
条边,则一定能匹配
所以第一行直接算出每个连通块内的边数就好了。
关键是要找到一个可行的算法来达到最优匹配。
然后我们可以发现,目标等价于最大化入度为偶数的点数。而无向边确定一个方向,等价于点放入两个极大团之一。那么搜索整个图,一开始所有点随便放入一个极大团,然后若点的入度为偶数,则可行。若点的入度为奇数,就可以把从该点父亲到该点的一条边反向。为了方便处理,直接把从根节点到该点的边反向。当然,取两次反等于没取。
然后记录最后每条边的方向。按照方案直接模拟匹配即可。
Code
#include<cstdio>
#include<cstring>
#include<algorithm>
#define fo(i,a,b) for(int i=a;i<=b;i++)
using namespace std;
const int N=20010,M=200010;
int tot,x[M],y[M],a[N],to[M*2],next[M*2],last[N];
bool bz[N],p[M];
void link(int u,int v)
{
to[++tot]=v;
next[tot]=last[u];
last[u]=tot;
}
int dfs(int v)
{
bz[v]=1;
int sum=a[v];
for(int i=last[v];i;i=next[i])
{
int u=to[i];
if(bz[u]) continue;
sum+=dfs(u);
}
return sum;
}
int dfs1(int v)
{
bz[v]=1;
int t=a[v]%2;
for(int i=last[v];i;i=next[i])
{
int u=to[i];
if(bz[u]) continue;
int k=dfs1(u);
t^=k,p[i/2]=k;
}
return t;
}
int main()
{
int n,m,ans=0;
scanf("%d %d",&n,&m);
tot=1;
fo(i,1,m)
{
scanf("%d %d",&x[i],&y[i]);
a[x[i]]++;
link(x[i],y[i]),link(y[i],x[i]);
}
fo(i,1,n)
if(!bz[i]) ans+=dfs(i)/2;
printf("%d\n",ans);
memset(bz,0,sizeof(bz));
fo(i,1,n)
if(!bz[i]) dfs1(i);
memset(last,0,sizeof(last));
tot=0;
fo(i,1,m)
if(p[i]) link(y[i],i);
else link(x[i],i);
fo(i,1,n)
{
int l=0;
for(int j=last[i];j;j=next[j])
{
if(!l) l=to[j];
else printf("%d %d\n",to[j],l),l=0;
}
}
return 0;
}