题意:给定N个字和M行他们之间的关系,要求输出他们的拓扑排序。此题采用边输入边检测的方式,如果发现环,就结束并输出当前行号;如果读取到当前行时,可以确定拓扑序列就输出,不管后面的输入(可能包含环路);如果到最后还是不能确定拓扑序列,就输出指定的字符串。

拓扑排序:对一个有向无环图(Directed Acyclic Graph简称DAG)G进行拓扑排序,是将G中所有顶点排成一个线性序列,使得图中任意一对顶点u和v,若 ∈E(G),则u在线性序列中出现在v之前。

分析:首先,拓扑排序的算法还是挺直观的。简单的拓扑排序算法:先找到任意一个没有入边的顶点,然后是显示该顶点,并将它和它的边一起从图中删除。然后,对图的其余部分应用同样的方法处理。如果找不到没有入边的顶点,说明存在回路。这里的实现过程由于要边输入边检测,所以要注意先一次性读取完输入。还要注意图不是强连通的,以及消去一个顶点后的图不是强连通图的这种情况。做的时候感觉要考虑很多情况,但做完了反而想不起那么多来。呵呵,看代码吧!

import java.util.Scanner;

public class Main {
static int n,m;
static int[][] a; //邻接矩阵
static int [] degree; //每个节点的入度
static boolean[] vis; //记录是否被访问
/**
* 拓扑排序,返回排序的序列,返回1说明存在环路。
* @return
*/
public static String topo(){
String s="";
for(int i=0;i<n && isIsolatedNode(i);i++){
int temp=findInDegreeZero();
if(temp!=-1){
//消去顶点后的图还是强连通就满足要求
if(countZeroDegree()==1)
s+=((char)(temp+65));
vis[temp]=true;
for(int j=0;j<n;j++)
if(a[temp][j]==1)
degree[j]--;
}else{
s="1";
break;
}
}
return s;
}
/**
* 读取入度
*/
public static void readInDegree(){
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
if(a[i][j]!=0 )
degree[j]++;
}
/**
* 由于采用邻接矩阵,所以要判断节点是否在要排序的N个节点中。利于查找入度为0的节点。
* @param num
* @return
*/
public static boolean isIsolatedNode(int num){
int sum=0;
for(int i=0;i<n;i++)
sum+=(a[num][i]+a[i][num]);
if(sum==0)
return false;
else
return true;
}
/**
* 计算入度为零的节点个数,超过一个则说明该图不是强连通图。
* 由于要优先判断是否存在环路,所以不能如果不是强连通图也要等判断是否存在环路。
* @return
*/
public static int countZeroDegree(){
int sum=0;
for(int i=0;i<n;i++)
if(degree[i]==0 && isIsolatedNode(i) && vis[i]==false){
sum++;
}
return sum;
}
/**
* 找到入度为0的节点,不存在就返回-1(存在环路)
* @return
*/
public static int findInDegreeZero(){
for(int i=0;i<n ;i++)
if(degree[i]==0 && vis[i]==false && isIsolatedNode(i)){
return i;
}
return -1;
} public static void main(String[] args) {
Scanner sc=new Scanner(System.in);
while(true){
n=sc.nextInt();
m=sc.nextInt();
if(n+m==0)
break;
a=new int[n][n];
String result="";
String[] str=new String[m+1];
for(int i=1; i<=m; i++)
str[i]=sc.next();
int i;
for(i=1; i<=m; i++){
vis=new boolean[n];
degree=new int[n];
a[str[i].charAt(0)-65][str[i].charAt(2)-65]=1;
readInDegree();
int beginZeroDegree=countZeroDegree();
result=topo();
//如果结果字符串长度和N相同,并且图是强连通的就输出序列
if(result.length()==n && beginZeroDegree==1){
System.out.println("Sorted sequence determined after "+i+" relations: "+result+".");
break;
//如果存在环路则输出
}else if(result.equals("1")){
System.out.println("Inconsistency found after "+i+" relations.");
break;
}
}
//不存在环路,到最后还是不能确定序列
if((i-1)==m && result.length()!=n)
System.out.println("Sorted sequence cannot be determined.");
}
}
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

拓扑排序 POJ 1094 Sorting It All Out的更多相关文章

  1. 拓扑排序 POJ 1049 Sorting It All Out

    题目传送门 /* 拓扑排序裸题:有三种情况: 1. 输入时发现与之前的矛盾,Inconsistency 2. 拓扑排序后,没有n个点(先判断cnt,即使一些点没有边连通,也应该是n,此时错误是有环): ...

  2. ACM: poj 1094 Sorting It All Out - 拓扑排序

    poj 1094 Sorting It All Out Time Limit:1000MS     Memory Limit:10000KB     64bit IO Format:%lld & ...

  3. 拓扑排序(Topological Sorting)

    一.什么是拓扑排序 在图论中,拓扑排序(Topological Sorting)是一个有向无环图(DAG, Directed Acyclic Graph)的所有顶点的线性序列.且该序列必须满足下面两个 ...

  4. poj 1094 Sorting It All Out (拓扑排序)

    http://poj.org/problem?id=1094 Sorting It All Out Time Limit: 1000MS   Memory Limit: 10000K Total Su ...

  5. POJ 1094 Sorting It All Out(拓扑排序+判环+拓扑路径唯一性确定)

    Sorting It All Out Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 39602   Accepted: 13 ...

  6. [ACM] POJ 1094 Sorting It All Out (拓扑排序)

    Sorting It All Out Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 26801   Accepted: 92 ...

  7. [ACM_模拟] POJ 1094 Sorting It All Out (拓扑排序+Floyd算法 判断关系是否矛盾或统一)

    Description An ascending sorted sequence of distinct values is one in which some form of a less-than ...

  8. POJ 1094 Sorting It All Out (拓扑排序,判断序列是否唯一,图是否有环)

    题意:给出n个字符,m对关系,让你输出三种情况:     1.若到第k行时,能判断出唯一的拓扑序列,则输出:         Sorted sequence determined after k re ...

  9. POJ 1094 Sorting It All Out (拓扑排序) - from lanshui_Yang

    Description An ascending sorted sequence of distinct values is one in which some form of a less-than ...

随机推荐

  1. 九度OJ 1343:城际公路网 (最小生成树)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:445 解决:178 题目描述: 为了加快城市之间的通行和物资流动速度,A国政府决定在其境内的N个大中型城市之间,增加修建K条公路.已知这N个 ...

  2. 【学员管理系统】0x04 数据库连接优化

    [学员管理系统]0x04  pymysql数据库连接优化 写在前面 项目详细需求参见:Django项目之[学员管理系统] 优化实现 把操作封装成函数 我们之前使用pymysql操作数据库的操作都是写死 ...

  3. go语言之接口二

    接口查询: 先来看如下的结构.结构体File实现了Read,Writer,Seek,Close的方法 type File struct{ } func (f *File) Read(buf []byt ...

  4. Excel 文件下载

    INCLUDE OLE2INCL * ALV输出   CALL FUNCTION 'REUSE_ALV_GRID_DISPLAY'     EXPORTING       I_CALLBACK_PRO ...

  5. linux 9 -- 交互式使用Bash Shell

    二十二. 交互式使用Bash Shell:     1.  用set命令设置bash的选项:     下面为set主要选项的列表及其表述: 选项名 开关缩写 描述 allexport -a 打开此开关 ...

  6. FOXMAIL提示容量满无法收邮件,清除旧邮件后还是无法收取,请问如何解决?

    FOXMAIL提示容量满无法收邮件,清除旧邮件后还是无法收取,请问如何解决? 2009-03-23 11:21包子燕  分类:网站使用 我清除了FOXMAIL所在的磁盘空间,共有12G,也删除了部分旧 ...

  7. STM32 ~ MDK环境下调试程序 HardFault_Handler 相关

    STM32出现HardFault_Handler故障的原因主要有两个方面: 1.内存溢出或者访问越界.这个需要自己写程序的时候规范代码,遇到了需要慢慢排查. 2.堆栈溢出.增加堆栈的大小. 出现问题时 ...

  8. CAS的实现Atomic类库

    atomic 原子(atomic)本意是"不能被进一步分割的最小粒子",而原子操作(atomic operation)意为"不可被中断的一个或一系列操作".在多 ...

  9. IImage--factory

    <?php /* 实例4 */ /* 使用工厂类解析图像工作 */ interface IImage { function getWidth(); function getHeight(); f ...

  10. 《python基础教程(第二版)》学习笔记 基础部分(第1章)

    <python基础教程(第二版)>学习笔记 基础部分(第1章)python常用的IDE:Windows: IDLE(gui), Eclipse+PyDev; Python(command ...