拓扑排序 POJ 1094 Sorting It All Out
题意:给定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的更多相关文章
- 拓扑排序 POJ 1049 Sorting It All Out
题目传送门 /* 拓扑排序裸题:有三种情况: 1. 输入时发现与之前的矛盾,Inconsistency 2. 拓扑排序后,没有n个点(先判断cnt,即使一些点没有边连通,也应该是n,此时错误是有环): ...
- ACM: poj 1094 Sorting It All Out - 拓扑排序
poj 1094 Sorting It All Out Time Limit:1000MS Memory Limit:10000KB 64bit IO Format:%lld & ...
- 拓扑排序(Topological Sorting)
一.什么是拓扑排序 在图论中,拓扑排序(Topological Sorting)是一个有向无环图(DAG, Directed Acyclic Graph)的所有顶点的线性序列.且该序列必须满足下面两个 ...
- poj 1094 Sorting It All Out (拓扑排序)
http://poj.org/problem?id=1094 Sorting It All Out Time Limit: 1000MS Memory Limit: 10000K Total Su ...
- POJ 1094 Sorting It All Out(拓扑排序+判环+拓扑路径唯一性确定)
Sorting It All Out Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 39602 Accepted: 13 ...
- [ACM] POJ 1094 Sorting It All Out (拓扑排序)
Sorting It All Out Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 26801 Accepted: 92 ...
- [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 ...
- POJ 1094 Sorting It All Out (拓扑排序,判断序列是否唯一,图是否有环)
题意:给出n个字符,m对关系,让你输出三种情况: 1.若到第k行时,能判断出唯一的拓扑序列,则输出: Sorted sequence determined after k re ...
- 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 ...
随机推荐
- WCF基础之序列化
wcf是基于消息进行通信的,这篇就是简单说下序列化引擎是如何将.net object转化为xml消息.一般情况下很少用到这些,你只需定义数据协定之类的或者指定相应的序列化引擎,然后设置相应的特性就好. ...
- ArcGIS API for JavaScript Bookmarks(书签)
说明:本篇博文介绍的是ArcGIS API for JavaScript中的 Bookmarks(书签) ,书签的作用是,把地图放大到一个地方 添加书签,书签名称可以和地图名称一直,单击标签 地图会定 ...
- 九度OJ 1283:第一个只出现一次的字符 (计数)
时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:1808 解决:997 题目描述: 在一个字符串(1<=字符串长度<=10000,全部由大写字母组成)中找到第一个只出现一次的字符 ...
- Python菜鸟之路:sqlalchemy/paramiko进阶
前言:ORM中的两种创建方式 数据库优先:指的是先创建数据库,包括表和字段的建立,然后根据数据库生成ORM的代码,它是先创建数据库,再创建相关程序代码 代码优先:就是先写代码,然后根据代码去生成数据库 ...
- [luogu3767]膜法
[luogu3767]膜法 luogu 神仙题 线段树分治+带权并查集 把每个操作看成点 首先这个操作的结构是一棵树 你发现每个点的对它的子树产生影响 我们可以想到用dfn序把它转成一段区间用线段树分 ...
- 教你使用SQL查询(1-12)
教你使用 Select 查询语句 (1) SELECT 语句基本语法简介 http://jimshu.blog.51cto.com/3171847/1363101(2) TOP 和 OFFSET 筛选 ...
- ionic添加scss
Setup Sass Automatically 在进行以下操作之前要确保node比较新,以便正确安装node-sass 或 改用cnpm install node-sass安装(淘宝源) $ ion ...
- 3.08课·········switch case及if else嵌套(日期格式)
switch case switch (n) { : break; : break; . . . case n: break; } 1.switch case必须与break一同使用,每一个case后 ...
- JS中如何获取<Select>中value和text的值
原文地址:JS中如何获取<Select>中value和text的值 html代码: <select id = "city" onchange="chan ...
- pulseaudio备注
参考http://www.ubuntu-tw.org/modules/newbb/viewtopic.php?viewmode=compact&topic_id=10102 Ubuntu 8. ...