题意还是比较容易理解的,关键要看到后面的:合条件的排名可能不是唯一的,此时要求输出时编号小的队伍在前;
思路:这道题就是拓扑排序的经典应用了,用队列做的考虑优先编号小的出队就可以了。
拓扑排序:
#include<iostream> #include<cstring> using namespace std; int map[502][502],flag[502],m,n,val[502]; void topsort() { int i, j, k=1; for(i=1; i<=n; i++) //有n个点,所以要找n次 { for(j=1; j<=n; j++) { if(flag[j]==0) //循环寻找度为0的点 { flag[j]--; val[k++]=j; for(int x=1; x<=n; x++) //删除指向该点的边 if(map[j][x]) flag[x]--; //使其度为0 break; } if(j>n) return ; } } } int main() { int i,j; while(cin>>n>>m) { memset(map,0,sizeof(map)); memset(flag,0,sizeof(flag)); //初始化所有的点的度为0 int a,b; for(i=1;i<=m;i++) { cin>>a>>b; if(!map[a][b]) { map[a][b]=1; flag[b]++; //度增加 } } topsort(); for(i=1;i<=n; i++) if(i!=n) cout<<val[i]<<" "; else cout<<val[i]<<endl; } return 0; }
优先队列优化后的代码:
#include<cstdio> #include<cstring> #include<queue> #include<vector> #include<algorithm> using namespace std; const int N=510+10; vector<int> g[N]; //邻接表 int du[N],val[N]; int n,m; void topsort() { int i; priority_queue<int, vector<int>, greater<int> > q;//定义一个优先队列,并定义编号小的优先出队 for(i=1;i<=n;i++) if(!du[i]) q.push(i); int cnt=1; while(!q.empty()) { int u=q.top(); q.pop(); val[cnt++]=u; for(i=0;i<g[u].size();i++) { int v=g[u][i]; du[v]--; if(!du[v]) q.push(v); } } } int main() { int i,j,u,v; while(scanf("%d %d",&n,&m)!=EOF) { memset(du,0,sizeof(du)); for(i=1;i<=n;i++)g[i].clear(); for(i=1;i<=m;i++) { scanf("%d %d",&u,&v); g[u].push_back(v); du[v]++; } topsort(); for(i=1;i<=n;i++) { if(i!=n) printf("%d ",val[i]); else printf("%d\n",val[i]); } } return 0; }
HDU 1285 确定比赛名次(拓扑排序模板),布布扣,bubuko.com
原文:http://blog.csdn.net/u012313382/article/details/38395339