逃生
题目链接
题目大意
现在有n个人排队,首先越小的优先级越高,其次有m个约束,每个约束都规定a必须在b前面。现在让你输出最后排队的状态。
题解
这一题初步看到可以确定是一个拓扑排序,而且小的在前面,一开始我没想太多直接就代码实现了,结果发现有问题:
比如这样一组数据
·1
3 1
3 1
我的程序结果是2 3 1,但是正解却是3 1 2。可以看到1的优先级应该是最高,然而在一开始,我们把度数为零的3 和2拿出来了,根据小的优先级高的原则,2先出,3再出,1最后。得到2 3 1。
我在发现按照小的优先级高的策略做不了的时候…我就把优先级反过来胡搞了一下发现输出的序列和正确的序列是反的,然后我反向输出结果,竟然对了…..
可是我并不知道为啥要这样写啊
代码
#include <iostream>
#include <cstring>
#include <cstdio>
#include <queue>
#include <vector>
using namespace std;
int T,n,m,d[30005],a,b,ans[30005],h;
vector<int> V[30005];
priority_queue<int> Q;
int main()
{
scanf("%d",&T);
while (T--)
{
memset(ans,0,sizeof(ans)); h=0;
memset(d,0,sizeof(d));
scanf("%d%d",&n,&m);
for (int i=0;i<=n;i++) V[i].clear();
for (int i=0;i<m;i++)
{
scanf("%d%d",&a,&b);
V[b].push_back(a);
d[a]++;
}
while (!Q.empty()) Q.pop();
for (int i=1;i<=n;i++) if (!d[i]) Q.push(i);
while(!Q.empty())
{
int v=Q.top();
Q.pop();
for (int i=0;i<V[v].size();i++)
{
int u=V[v][i];
d[u]--;
if (!d[u]) Q.push(u);
}
ans[h++]=v;
}
for (int i=h-1;i>0;i--) printf("%d ",ans[i]);
printf("%d\n",ans[0]);
}
return 0;
}