HDU1285 确定比赛名次【拓扑排序】【优先队列】

确定比赛名次


Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)
Total Submission(s): 13215    Accepted Submission(s): 5291

Problem Description
有N个比赛队(1<=N<=500),编号依次为1,2,3,。。。。,N进行比赛,比赛结束后,裁判委员会要将所有参赛队伍从前往后依次排名,但现在裁判委员会不能直接获得每个队的比赛成绩,只知道每场比赛的结果,即P1赢P2,用P1,P2表示,排名时P1在P2之前。现在请你编程序确定排名。
 
Input
输入有若干组,每组中的第一行为二个数N(1<=N<=500),M;其中N表示队伍的个数,M表示接着有M行的输入数据。接下来的M行数据中,每行也有两个整数P1,P2表示即P1队赢了P2队。
 
Output
给出一个符合要求的排名。输出时队伍号之间有空格,最后一名后面没有空格。

其他说明:符合条件的排名可能不是唯一的,此时要求输出时编号小的队伍在前;输入数据保证是正确的,即输入数据确保一定能有一个符合要求的排名。
 
Sample Input
4 3
1 2
2 3
4 3
 
Sample Output

1 2 4 3


思路:因为要满足字典序的拓扑排序,所以用了STL中的优先队列。

priority_queue<int,vector<int>, greater<int> > Q;

实现了权值小的优先级高,取出的时候保证序号是队列中最小的。

其他的和一般的拓扑排序无区别。


#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<queue>
#include<vector>
using namespace std;
const int MAXN = 550;
const int MAXM = 25000;

int head[MAXN],indegree[MAXN],ans[MAXN],N,M;
struct EdgeNode
{
    int to;
    int w;
    int next;
};
EdgeNode Edges[MAXM];

int toposort()
{
    priority_queue<int,vector<int>, greater<int> > Q;
    for(int i = 1; i <= N; i++)
    {
        if(indegree[i]==0)
            Q.push(i);
    }
    int id = 0;
    while(!Q.empty())
    {
        int u;
        ans[id++] = u = Q.top();
        Q.pop();
        for(int k = head[u]; k != -1; k = Edges[k].next)
        {
            indegree[Edges[k].to]--;
            if(!indegree[Edges[k].to])
            {
                Q.push(Edges[k].to);
            }
        }
    }
    if(id == N)
    {
        for(int i = 0; i < id; i++)
        {
            if(i != id-1)
                cout << ans[i] << " ";
            else
                cout << ans[i] << endl;
        }
    }
    else
        return 0;
}
int main()
{
    int x,y;
    while(cin >> N >> M)
    {
        memset(ans,0,sizeof(ans));
        memset(head,-1,sizeof(head));
        memset(Edges,0,sizeof(Edges));
        for(int i = 0; i < M; i++)
        {
            cin >> x >> y;
            Edges[i].to = y;
            Edges[i].w = 1;
            Edges[i].next = head[x];
            head[x] = i;
            indegree[y]++;
        }
        toposort();
    }

    return 0;
}



你可能感兴趣的:(HDU1285 确定比赛名次【拓扑排序】【优先队列】)