▶ 书中第四章部分程序,包括在加上自己补充的代码,两种拓扑排序的方法

● 拓扑排序 1

 package package01;

 import edu.princeton.cs.algs4.Digraph;
import edu.princeton.cs.algs4.SymbolDigraph;
import edu.princeton.cs.algs4.DirectedCycle;
import edu.princeton.cs.algs4.DepthFirstOrder;
import edu.princeton.cs.algs4.EdgeWeightedDigraph;
import edu.princeton.cs.algs4.EdgeWeightedDirectedCycle; public class class01
{
private Iterable<Integer> order; // 拓扑排序的结果
private int[] rank; // 顶点 v 在拓扑排序中的序号为 rank[v] public class01(Digraph G) // 从有向图生成拓扑排序
{
DirectedCycle finder = new DirectedCycle(G); // 存在环则不能排序
if (!finder.hasCycle())
{
DepthFirstOrder dfs = new DepthFirstOrder(G); // 做 G 的深度优先搜索
order = dfs.reversePost(); // 取逆后序依次标号
rank = new int[G.V()];
int i = 0;
for (int v : order)
rank[v] = i++;
}
} public class01(EdgeWeightedDigraph G) // 从加权边有向图生成拓扑排序(算法一样,只是数据结构不同)
{
EdgeWeightedDirectedCycle finder = new EdgeWeightedDirectedCycle(G);
if (!finder.hasCycle())
{
DepthFirstOrder dfs = new DepthFirstOrder(G);
order = dfs.reversePost();
rank = new int[G.V()];
int i = 0;
for (int v : order)
rank[v] = i++;
}
} public Iterable<Integer> order()
{
return order;
} public boolean hasOrder()
{
return order != null;
} public int rank(int v)
{
return hasOrder() ? rank[v] : -1;
} public static void main(String[] args)
{
String filename = args[0];
String delimiter = args[1]; // 分隔符
SymbolDigraph sg = new SymbolDigraph(filename, delimiter);
class01 topological = new class01(sg.digraph());
for (int v : topological.order())
System.out.println(sg.nameOf(v));
}
}

● 拓扑排序 2

 package package01;

 import edu.princeton.cs.algs4.StdOut;
import edu.princeton.cs.algs4.StdRandom;
import edu.princeton.cs.algs4.DigraphGenerator;
import edu.princeton.cs.algs4.Digraph;
import edu.princeton.cs.algs4.Queue;
import edu.princeton.cs.algs4.DirectedEdge;
import edu.princeton.cs.algs4.EdgeWeightedDigraph; public class class01
{
private Queue<Integer> order;
private int[] rank; public class01(Digraph G)
{
order = new Queue<Integer>();
rank = new int[G.V()];
int[] indegree = new int[G.V()];
for (int v = 0; v < G.V(); v++)
indegree[v] = G.indegree(v);
int count = 0;
Queue<Integer> queue = new Queue<Integer>();
for (int v = 0; v < G.V(); v++) // 收集所有没有前提条件的顶点
{
if (indegree[v] == 0)
queue.enqueue(v);
}
for (; !queue.isEmpty();)
{
int v = queue.dequeue();
order.enqueue(v); // 事件 v 完成,将其放入输出队列, 并给一个序号
rank[v] = count++;
for (int w : G.adj(v)) // 所有紧接着 v 的事件的前提条件减少 1
{
indegree[w]--;
if (indegree[w] == 0) // 收集此时没有前提条件的事件
queue.enqueue(w);
}
}
if (count != G.V()) // 遍历结束,还有顶点有入度,说明存在环
order = null;
} public class01(EdgeWeightedDigraph G)
{
order = new Queue<Integer>();
rank = new int[G.V()];
int[] indegree = new int[G.V()];
for (int v = 0; v < G.V(); v++)
indegree[v] = G.indegree(v);
int count = 0;
Queue<Integer> queue = new Queue<Integer>();
for (int v = 0; v < G.V(); v++)
{
if (indegree[v] == 0)
queue.enqueue(v);
}
for (; !queue.isEmpty();)
{
int v = queue.dequeue();
order.enqueue(v);
rank[v] = count++;
for (DirectedEdge e : G.adj(v))
{
int w = e.to();
indegree[w]--;
if (indegree[w] == 0)
queue.enqueue(w);
}
}
if (count != G.V())
order = null;
} public Iterable<Integer> order()
{
return order;
} public boolean hasOrder()
{
return order != null;
} public int rank(int v)
{
return hasOrder() ? rank[v] : -1;
} public static void main(String[] args)
{
int V = Integer.parseInt(args[0]); // 生成DAG G(V,E),再添加 F 条边
int E = Integer.parseInt(args[1]);
int F = Integer.parseInt(args[2]);
Digraph G1 = DigraphGenerator.dag(V, E); // G1 是无边圈的
EdgeWeightedDigraph G2 = new EdgeWeightedDigraph(V);// G2 有边权的
for (int v = 0; v < G1.V(); v++)
{
for (int w : G1.adj(v))
G2.addEdge(new DirectedEdge(v, w, 0.0));
}
for (int i = 0; i < F; i++)
{
int v = StdRandom.uniform(V);
int w = StdRandom.uniform(V);
G1.addEdge(v, w);
G2.addEdge(new DirectedEdge(v, w, 0.0));
}
StdOut.println(G1);
StdOut.println();
StdOut.println(G2);
class01 topological1 = new class01(G1); // 分别计算 G1 和 G2 的
if (!topological1.hasOrder())
StdOut.println("Not a DAG");
else
{
StdOut.print("Topological order: ");
for (int v : topological1.order())
StdOut.print(v + " ");
StdOut.println();
}
class01 topological2 = new class01(G2);
if (!topological2.hasOrder())
StdOut.println("Not a DAG");
else
{
StdOut.print("Topological order: ");
for (int v : topological2.order())
StdOut.print(v + " ");
StdOut.println();
}
}
}

《算法》第四章部分程序 part 9的更多相关文章

  1. 《算法》第四章部分程序 part 19

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,有边权有向图的邻接矩阵,FloydWarshall 算法可能含负环的有边权有向图任意两点之间的最短路径 ● 有边权有向图的邻接矩阵 package p ...

  2. 《算法》第四章部分程序 part 18

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,在有权有向图中寻找环,Bellman - Ford 算法求最短路径,套汇算法 ● 在有权有向图中寻找环 package package01; impo ...

  3. 《算法》第四章部分程序 part 16

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,Dijkstra 算法求有向 / 无向图最短路径,以及所有顶点对之间的最短路径 ● Dijkstra 算法求有向图最短路径 package packa ...

  4. 《算法》第四章部分程序 part 15

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,Kruskal 算法和 Boruvka 算法求最小生成树 ● Kruskal 算法求最小生成树 package package01; import e ...

  5. 《算法》第四章部分程序 part 14

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,两种 Prim 算法求最小生成树 ● 简单 Prim 算法求最小生成树 package package01; import edu.princeton ...

  6. 《算法》第四章部分程序 part 10

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,包括无向图连通分量,Kosaraju - Sharir 算法.Tarjan 算法.Gabow 算法计算有向图的强连通分量 ● 无向图连通分量 pack ...

  7. 《算法》第四章部分程序 part 17

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,无环图最短 / 最长路径通用程序,关键路径方法(critical path method)解决任务调度问题 ● 无环图最短 / 最长路径通用程序 pa ...

  8. 《算法》第四章部分程序 part 13

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,图的前序.后序和逆后续遍历,以及传递闭包 ● 图的前序.后序和逆后续遍历 package package01; import edu.princeto ...

  9. 《算法》第四章部分程序 part 12

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,图的几种补充数据结构,包括无向 / 有向符号图,有权边结构,有边权有向图 ● 无向符号图 package package01; import edu. ...

随机推荐

  1. 美剧黑名单的插曲《Jolene》

    网易上有Slowdown 版本.<Jolene>Dolly Parton

  2. Python关于self用法重点分析

    在介绍Python的self用法之前,先来介绍下Python中的类和实例…… 我们知道,面向对象最重要的概念就是类(class)和实例(instance),类是抽象的模板,比如学生这个抽象的事物,可以 ...

  3. Git将本地库内容推送到远程

    本地库与远程库的交互 1 .将本地库的内容推送到远程库 A.创建一个本地仓库 $ mkdir gitdemo B.初始化本地仓库 $ git init C.项目根目录下创建 .gitignore 文件 ...

  4. quartz里job不执行的解决方案(并发量太低原因)

    这里写链接内容 使用框架spring3+quartz1.8 生产环境中碰到会有job一直不执行的情况,后来分析是因为quartz中线程总数太少,而项目中所有的job都是并发执行的就会导致当到达时间节点 ...

  5. 机器学习-Python中训练模型的保存和再使用

    模型保存 BP:model.save(save_dir) SVM: from sklearn.externals import joblib joblib.dump(clf, save_dir) 模型 ...

  6. bzoj4183: tree

    Description 自底向上模拟,原地操作以节省空间 #include<bits/stdc++.h> unsigned n,a,b,c,v[],mx,ans=; int main(){ ...

  7. SSM的配置文件

    Mybatis: SqlMapConfig.xml,配置了数据源,连接池,事务,加载sql映射文件(pojo),sqlsessionFactory对象,配置到spring容器中,mapeer代理对象或 ...

  8. ueditor 正在读取目录

    ueditor 版本为1.3.6  项目版本为2.0 引用 <script src="../ueditor/ueditor.config.js" type="tex ...

  9. Unreal Engine 4 Smear Frame效果的实现与分析

    转自:http://www.52vr.com/article-868-1.html 这篇文章介绍了类似守望先锋中的帧转移模糊(Smear Frame)效果.   该效果由Jan Kaluza实现,本博 ...

  10. (error) MOVED 5798 172.17.0.3:6379

    登录没有启动集群模式(即缺少了那个"-c"): redis-cli -c -h yourhost -p yourpost