JZOJ 4376【GDOI2016模拟3.9】染色配对

Description

JZOJ 4376【GDOI2016模拟3.9】染色配对_第1张图片
这里写图片描述
这里写图片描述

Analysis

由于题目的特性,我们把极大团看成点,把点看成边。
然后会形成多个连通块。
对于每个连通块,若其中有 m 条边,则一定能匹配 m2 次。因为一次匹配要两条边。
所以第一行直接算出每个连通块内的边数就好了。
关键是要找到一个可行的算法来达到最优匹配。
然后我们可以发现,目标等价于最大化入度为偶数的点数。而无向边确定一个方向,等价于点放入两个极大团之一。那么搜索整个图,一开始所有点随便放入一个极大团,然后若点的入度为偶数,则可行。若点的入度为奇数,就可以把从该点父亲到该点的一条边反向。为了方便处理,直接把从根节点到该点的边反向。当然,取两次反等于没取。
然后记录最后每条边的方向。按照方案直接模拟匹配即可。

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;
}

你可能感兴趣的:(贪心)