题目连接: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;
}
}
}