HDU 2647 Reward(反向拓扑)

来源:互联网 发布:重复犯错歌词 知乎 编辑:程序博客网 时间:2024/06/05 12:48

这题给的内存忒小了,不能用邻接矩阵存,只能用邻接表。然后一个反向的拓扑排序就好了,但最后的结果是每个点中存的信息(我应该比几个人的钱多)的和再加上888*总共有多少人。

看了很多人的题解,发现主要会出现错误的地方就是没有反向拓扑排序,或者是反向拓扑排序以后用等差公式求和(假设有1->3,2->4->3的情况,拓扑排序以后得到是一个线性的序列,那么就会认为1->2或者是2->1,但实际上对点1和点2两个人的话,他们得到的钱数应该是相等的)。

#include <iostream>#include <cstdio>#include <cstring>#include <queue>#include <vector>using namespace std;typedef pair<int,int> p;queue <p> q;#define maxn 10010int point[maxn];vector <int> graph[maxn];int indegree[maxn];int m,n;bool toposort(){for(int i=1;i<=m;i++){if(indegree[i]==0){q.push(make_pair(i,0));}}int cnt=0;while(!q.empty()){p pp=q.front();q.pop();cnt++;int tmp=pp.first;int id=pp.second;for(int i=0;i<graph[tmp].size();i++){int tmp1=graph[tmp][i];indegree[tmp1]--;point[tmp1]=max(id+1,point[tmp1]);if(indegree[tmp1]==0){q.push(make_pair(tmp1,point[tmp1]));}}}if(cnt==m)return true;elsereturn false;}int main(){while(scanf("%d %d",&m,&n)!=EOF){memset(indegree,0,sizeof(indegree));memset(point,0,sizeof(point));for(int i=1;i<=m;i++){    graph[i].clear();}for(int i=1;i<=n;i++){int u,v;scanf("%d %d",&u,&v);graph[v].push_back(u);indegree[u]++;}if(toposort()){int sum=0;for(int i=1;i<=m;i++){sum+=point[i];}cout<<sum+m*888<<endl;}else{cout<<"-1"<<endl;}}return 0;}

0 0
原创粉丝点击