nyoj 367 Reward

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


你可能感兴趣的:(nyoj 367 Reward)