题目描述
有N个士兵(1≤N≤26),编号依次为A,B,C,…,队列训练时,指挥官要把一些士兵从高到矮一次排成一行,但现在指挥官不能直接获得每个人的身高信息,只能获得“P1比P2高”这样的比较结果(P1、P2∈A,B,C,…,Z,记为 P1>P2),如”A>B”表示A比B高。
请编一程序,根据所得到的比较结果求出一种符合条件的排队方案。
(注:比较结果中没有涉及的士兵不参加排队)
输入格式
比较结果从文本文件中读入(文件由键盘输入),每个比较结果在文本文件中占一行。
输出格式
若输入数据无解,打印“No Answer!”信息,否则从高到矮依次输出每一个士兵的编号,中间无分割符,并把结果写入文本文件中,文件由键盘输入:
输入样例
A>B
B>D
F>D
输出样例
AFBD
分析
本题是道裸拓扑问题,如果在进行拓扑排序的过程结束后发现有的点的入度仍不为0,则说明图中存在环,直接输出"No Answer",否则说明是一个拓扑图,直接输出拓扑序。
C++ 代码
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
int T,n,d[30],co;
bool st[30];
char s[10],ans[100];
vector<int> v[26];
int main()
{
while(scanf("%s",s) !=EOF)
{
int a=s[0]-'A',b=s[2]-'A';
st[a]=st[b]=1;
v[a].push_back(b);
d[b]++;
}
queue<int> q;
for(int i=0;i<26;i++) //初始先将入度为0的点加入到队列中
{
if(!d[i] && st[i])
q.push(i);
}
while(q.size()) //拓扑排序
{
auto t=q.front();
q.pop();
ans[co++]=char(t+'A');
int len=v[t].size();
for(int i=0;i<len;i++)
{
if(!(--d[v[t][i]])) q.push(v[t][i]);
}
}
for(int i=0;i<26;i++) //判环
{
if(d[i])
{
puts("No Answer!");
return 0;
}
}
for(int i=0;i<co;i++)
{
cout<<ans[i];
}
return 0;
}