码迷,mamicode.com
首页 > 其他好文 > 详细

ZOJ3795_Grouping

时间:2014-08-06 14:07:41      阅读:250      评论:0      收藏:0      [点我收藏+]

标签:style   blog   color   os   io   for   问题   ar   

告诉你某些人的年龄大小关系,问你把所有的人分成若干个组,最少需要多少组,使得组内任意两个人的年龄不可比。

首先考虑特殊情况,如果所有年龄关系构成了一个环,那么这个环中所有人的年龄都是相等,也就是可比的。

同时所有其他的与这个环中任意一个点相连的任意一个环或者点都是可比的。

如果两个点或者环,无法处在同一条路径上,那么这两个点和环就是不可比的。

于是算法就出来了。

对于每一个强连通分量,我们将其缩为一个点,点权为这个连通分量中的点数。这样就相当于我们找一条权值最大的路径就好了。

由于缩点后一定是一个有向无环图,那么可以通过dp或者记忆化搜索解决问题。

 

 

召唤代码君:

 

 

#include <iostream>
#include <cstring>
#include <cstdio>
#include <vector>
#define maxn 600600
using namespace std;

vector<int> V[maxn];
int to[maxn],next[maxn],first[maxn],edge;
int d[maxn],f[maxn],low[maxn],belong[maxn],ans;
int stack[maxn],top;
int n,m,sccnum,dfs_clock;

void _init()
{
    edge=ans=0;
    top=sccnum=0;
    memset(first,-1,sizeof(int)*(n+2));
    memset(d,0,sizeof(int)*(n+2));
    memset(belong,0,sizeof(int)*(n+2));
}

void addedge(int U,int V)
{
    if (U==V) return;
    to[edge]=V,next[edge]=first[U],first[U]=edge++;
}

void dfs(int cur)
{
    d[cur]=low[cur]=++dfs_clock;
    stack[++top]=cur;
    for (int i=first[cur]; i!=-1; i=next[i])
        if (!d[to[i]]){
            dfs(to[i]);
            low[cur]=min(low[cur],low[to[i]]);
        }
        else if (!belong[to[i]]) low[cur]=min(low[cur],d[to[i]]);
    if (low[cur]==d[cur]){
        V[++sccnum].clear();
        f[sccnum]=0;
        for (;;){
            int x=stack[top--];
            belong[x]=sccnum;
            f[sccnum]++;
            if (x==cur) break;
        }
    }
}

int get(int x)
{
    if (d[x]!=-1) return d[x];
    d[x]=0;
    for (unsigned i=0; i<V[x].size(); i++)
        d[x]=max(get(V[x][i]),d[x]);
    return d[x]=d[x]+f[x];
}

int main()
{
    int UU,VV;
    while (scanf("%d%d",&n,&m)!=EOF){
        _init();
        while (m--){
            scanf("%d%d",&UU,&VV);
            addedge(UU,VV);
        }
        for (int i=1; i<=n; i++)
            if (!d[i]) dfs(i);
        for (int i=1; i<=n; i++)
            for (int j=first[i]; j!=-1; j=next[j])
                if (belong[i]!=belong[to[j]])
                    V[belong[i]].push_back(belong[to[j]]);
        memset(d,-1,sizeof(int)*(sccnum+2));
        for (int i=1; i<=sccnum; i++)
            ans=max(ans,get(i));
        printf("%d\n",ans);
    }
    return 0;
}

 

ZOJ3795_Grouping,布布扣,bubuko.com

ZOJ3795_Grouping

标签:style   blog   color   os   io   for   问题   ar   

原文地址:http://www.cnblogs.com/Canon-CSU/p/3894132.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!