题目链接: 最短工期 (25分)
拓扑排序
只适用于有向无环图
有向无环图一定有拓扑排序
核心算法:
- 把入度为0的点存放在队列里面
- 删除该顶点连接的所有边
解题步骤:
- 记录每个点的入度大小
- 把入度为0的点放入队列中去
- 用一个dist[]数组存放路径的大小,用cnt记录顶点个数
- 删除该顶点连接的所有边,即每条边对应的点的入度-1
- 再次把入度为0的点存放在队列里面
- 更新dist[]数组的大小,存放路径最长的
如果记录的顶点个数小于该图中的顶点个数,则该图中存在回路,不能实现拓扑排序
代码如下:
#include<iostream> #include<vector> #include<algorithm> #include<queue> #include<cstring> using namespace std; const int N = 110; #define mm(a,x) memset(a,x,sizeof a) #define inf 0x3f3f3f3f int mp[N][N],d[N],dist[N]; int n,m; int ans; int topsort(){ queue<int > q; int cnt = 0; for(int i = 0; i < n; i ++ ){ if(d[i] == 0) q.push(i); } while(q.size()){ int t = q.front(); q.pop(); cnt ++; if(dist[t] > ans) ans = dist[t]; for(int i = 0; i < n; i ++ ){ if(mp[t][i] != inf){ d[i] --; if(d[i] == 0) q.push(i); if(dist[i] < dist[t] + mp[t][i]){ dist[i] = dist[t] + mp[t][i]; } } } } if(cnt < n) return 0; else return ans; } int main(){ cin >> n >> m; mm(mp,inf); for(int i = 0; i < m; i ++ ){ int a,b,c; cin >> a >> b >> c; mp[a][b] = c; d[b] ++; } int t = topsort(); if(t == 0) cout<<"Impossible\n"; else cout<<t; return 0; }