nyoj 367 Reward

来源:互联网 发布:淘宝盗图被投诉怎么办 编辑:程序博客网 时间:2024/06/14 02:46

题目连接:http://acm.nyist.net/JudgeOnline/problem.php?pid=367

题意:一个老板一个给公司员工发奖金,每个人至少888,但是员工之间会有争议,为了能消除员工之间的争议,不同员之间的奖金可能就会不同。老板希望以最少的奖金解决争议。

解题思路:一开始我以为是并查集,可是提交一直WA,后来一想是拓扑排序。先找到入度为零的点,然后出度对应的点的入读减1(此过程会再次产生入读为0的点),如果出度和入读都为零,这可将这个点删去(同时更新该员工的工资),同时统计删除的点的个数,如果最后删除点的额个数不等与员工数,那么就无法消除争议。


 #include<iostream>#include<algorithm>#include<cstring>#include<vector>#include<queue>using namespace std;struct A{int x,y;}a[1010];vector<int> s[1010];   //保存各个点的邻接点int b[1010];           //保存各个员工的工资int main(){int m,n;while(cin>>m>>n){memset(b,888,sizeof(b));for(int i=1;i<=m;i++){a[i].x=0;a[i].y=0;s[i].clear();}for(int i=0;i<n;i++){int x,y;cin>>x>>y;a[x].x++;a[y].y++;s[y].push_back(x);}queue<int> r;for(int i=1;i<=m;i++){if(a[i].x==0)r.push(i);   //找到入读为0 的点}if(r.empty())cout<<-1<<endl;     //如果不存在入读为0 的点,说明无法消除争议else{int sum=888*m;while(!r.empty()){int c=r.front();for(int i=0;i<s[c].size();i++){int d=s[c][i];a[c].y--;a[d].x--;if(a[d].x==0&&a[d].y!=0)r.push(d);if(b[d]<=b[c]){sum+=b[c]+1-b[d];b[d]=b[c]+1;}}r.pop();}int flat=0;for(int i=1;i<=m;i++){if(a[i].x!=0||a[i].y!=0){flat=1;break;}}if(flat)cout<<-1<<endl;else cout<<sum<<endl;}}}