首页 > 其他 > 详细

bzoj 1006 MCS算法

时间:2014-02-24 11:28:06      阅读:357      评论:0      收藏:0      [点我收藏+]

  根据陈丹琪的论文弦图与区间图,求出弦图的完美消除序列之后,反向给每个点染可以染的最小的颜色,这样可以使用最少的颜色染色,染色的方案即为队伍数。

  那么我们需要求该图的完美消除序列,使用MCS算法,从后向前求完美消除序列,我们设size[i]为i这个点的标号,表示该点与多少个以标记的点相连,选取标号最大且i最大的点(即n)为序列的第n个,然后更新与i相连的点的标号,重复n次,即可得到弦图的完美消除序列,使用动态链表维护可以使时间复杂度达到O(m+n),暴力循环也有n^2,这道题暴力即可过。

bubuko.com,布布扣
/**************************************************************
    Problem: 1006
    User: BLADEVIL
    Language: C++
    Result: Accepted
    Time:1720 ms
    Memory:20492 kb
****************************************************************/
 
//By BLADEVIL
#include <cstdio>
#include <cstring>
#include <algorithm>
#define maxn 10010
#define maxm 1000010
 
using namespace std;
 
int n,m,l,ans;
int peo[maxn],size[maxn],flag[maxn],color[maxn];
int last[maxm],other[maxm<<1],pre[maxm<<1];
 
void connect(int x,int y){
    pre[++l]=last[x];
    last[x]=l;
    other[l]=y;
}
 
int main(){
    scanf("%d%d",&n,&m);
    for (int i=1;i<=m;i++){
        int x,y;
        scanf("%d%d",&x,&y);

        connect(x,y); connect(y,x);
    }
    for (int i=n;i;i--){
        int cur=0;
        for (int j=1;j<=n;j++) cur=!flag[j]&&size[j]>=size[cur]?j:cur;
        flag[cur]=1; peo[i]=cur;
        for (int p=last[cur];p;p=pre[p]) size[other[p]]++;
    }
//for (int i=1;i<=n;i++) printf("%d ",peo[i]); printf("\n");
    memset(flag,0,sizeof flag);
    for (int i=n;i;i--){
        for (int p=last[peo[i]];p;p=pre[p]) flag[color[other[p]]]=i;
        for (color[peo[i]]=1;flag[color[peo[i]]]==i;color[peo[i]]++);
        ans=max(ans,color[peo[i]]);
    }
    printf("%d\n",ans);
    return 0;
}
bubuko.com,布布扣

bzoj 1006 MCS算法

原文:http://www.cnblogs.com/BLADEVIL/p/3562193.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!