zcmu 2103 士兵排队问题(拓扑排序+优先队列)

【题目】

2103: 士兵排队问题

Time Limit: 1 Sec   Memory Limit: 128 MB
Submit: 22   Solved: 7
[ Submit][ Status][ Web Board]

Description

有N个士兵(1≤N≤26),编号依次为 A,B,C,…,队列训练时,指挥官要把一些士兵从高到矮一次排成一行,但现在指挥官不能直接获得每个人的身高信息,只能获得“P1比P2高”这样的比较 结果(P1、P2∈A,B,C,…,Z,记为 P1>P2),如”A>B”表示A比B高。
请编一程序,根据所得到的比较结果求出一种字典序最小的排队方案。
(注:比较结果中没有涉及的士兵不参加排队)

Input

比较结果从文本文件中读入(文件由键盘输入),每个比较结果在文本文件中占一行。

Output

若输入数据无解,打印“No Answer!”信息,否则从高到矮一次输出每一个士兵的编号,中间无分割符,并把结果写入文本文件中,文件由键盘输入:

Sample Input

A>B
B>D
F>D

Sample Output

ABFD 

【题解】

啦啦啦,之前挂过一道基本一毛一样的题,现在又回来了,争取讲的详细些。

题意就是根据局部顺序推整体的顺序,自然想到拓扑排序啦。比较后的可能情况有多种,要求字典序最小就只有一种啦。字典序最小,就要用优先队列啦,自动排序,天然的外挂呀~省事的很。

下面是我做过最全注释的代码!!!

【代码】

#include<bits/stdc++.h>
using namespace std;
const int maxn=100; //因为把字符转换成ASCII码肯定小于100,图省事
vector <int> edge[maxn]; 
int in[maxn]={0}; //入度
int f[maxn]={0}; //标记状态,是否出现过
int ans[maxn]={0}; //保存答案
priority_queue< int,vector <int> ,greater <int> >q; //声明一个优先队列q
void init() //初始化
{
    for(int i=1;i<100;i++)
        edge[i].clear();
    while(!q.empty())
         q.pop();
}
int main()
{
    init();
    char a,b;
    int c=0,i,t=0,k;
    while(~scanf(" %c>%c",&a,&b)) //空格消除缓存区干扰
    {
        if(f[a]==0)
            f[a]=1,c++;
        if(f[b]==0)
            f[b]=1,c++;
        in[b]++;
        edge[a].push_back(b);
    }
    for(i='A';i<='Z';i++)
        if(f[i]&&in[i]==0) //找开始拓扑排序的入度为0的点
           q.push(i);
    while(!q.empty()) //拓扑排序
    {
        k=q.top();
        ans[t++]=k;
        q.pop();
        for(i=0;i<edge[k].size();i++)
        {
            in[edge[k][i]]--;
            if(in[edge[k][i]]==0)
                q.push(edge[k][i]);
        }
    }
    if(t==c) //判断是否可行
    {
        for(i=0;i<t;i++)
            printf("%c",ans[i]);
    }
    else
        printf("No Answer!");
    printf("\n");
    return 0;
}

猜你喜欢

转载自blog.csdn.net/qq_41117236/article/details/80766801