▶ 书中第四章部分程序,包括在加上自己补充的代码,两种 Prim 算法求最小生成树

● 简单 Prim 算法求最小生成树

 package package01;

 import edu.princeton.cs.algs4.In;
import edu.princeton.cs.algs4.StdOut;
import edu.princeton.cs.algs4.Edge;
import edu.princeton.cs.algs4.EdgeWeightedGraph;
import edu.princeton.cs.algs4.Queue;
import edu.princeton.cs.algs4.MinPQ; public class class01
{
private static final double FLOATING_POINT_EPSILON = 1E-12; private boolean[] marked; // 顶点是否在生成树中
private double weight; // 生成树的权值和
private Queue<Edge> mst; // 生成树包含的边
private MinPQ<Edge> pq; // 搜索队列 public class01(EdgeWeightedGraph G)
{
marked = new boolean[G.V()];
mst = new Queue<Edge>();
pq = new MinPQ<Edge>();
for (int v = 0; v < G.V(); v++) // 对每个没有遍历的节点都使用 prim
{
if (!marked[v])
prim(G, v);
}
} private void prim(EdgeWeightedGraph G, int s)
{
for (scan(G, s); !pq.isEmpty();)
{
Edge e = pq.delMin(); // 取出权值最小的边
int v = e.either(), w = e.other(v);
if (marked[v] && marked[w]) // 若该边两端都遍历过,不要(由于 scan,v 与 w 之一肯定被遍历过)
continue;
mst.enqueue(e); // 将权值最小的边加入生成树
weight += e.weight(); // 更新权值和
if (!marked[v]) // 从新边的新顶点继续收集新的边
scan(G, v);
if (!marked[w])
scan(G, w);
}
} private void scan(EdgeWeightedGraph G, int v) // 将一端为 v、另一端没有遍历过的边放入队列中
{
marked[v] = true;
for (Edge e : G.adj(v))
{
if (!marked[e.other(v)])
pq.insert(e);
}
} public Iterable<Edge> edges()
{
return mst;
} public double weight()
{
return weight;
} public static void main(String[] args)
{
In in = new In(args[0]);
EdgeWeightedGraph G = new EdgeWeightedGraph(in);
class01 mst = new class01(G);
for (Edge e : mst.edges())
StdOut.println(e);
StdOut.printf("%.5f\n", mst.weight());
}
}

● 改进,使用索引最小优先队列来建立搜索队列,记录(起点到)每个顶点的距离来判断是否将新边加入生成树

 package package01;

 import edu.princeton.cs.algs4.In;
import edu.princeton.cs.algs4.StdOut;
import edu.princeton.cs.algs4.Edge;
import edu.princeton.cs.algs4.EdgeWeightedGraph;
import edu.princeton.cs.algs4.Queue;
import edu.princeton.cs.algs4.IndexMinPQ; public class class01
{
private static final double FLOATING_POINT_EPSILON = 1E-12; private boolean[] marked;
private Edge[] edgeTo; // 除了搜索起始顶点,新加入每条边对应一个顶点,顶点 v 对应的边是 edgeTo[v]
private double[] distTo; // 生成树到每个顶点的距离,用于衡量新边是否值得加入生成树
private IndexMinPQ<Double> pq;// 搜索队列 public class01(EdgeWeightedGraph G)
{
marked = new boolean[G.V()];
edgeTo = new Edge[G.V()];
distTo = new double[G.V()];
pq = new IndexMinPQ<Double>(G.V());
for (int v = 0; v < G.V(); v++)
distTo[v] = Double.POSITIVE_INFINITY;
for (int v = 0; v < G.V(); v++)
{
if (!marked[v])
prim(G, v);
}
} private void prim(EdgeWeightedGraph G, int s)
{
distTo[s] = 0.0; // 搜索起点对应的距离为 0
for (pq.insert(s, distTo[s]); !pq.isEmpty();)
{
int v = pq.delMin(); // 每次取距离最小的顶点来开花(防止同一个顶点可以有对多条边连到树上)
scan(G, v); // 注意 scan 只负责在给定的顶点上开花,不负责递归
}
} private void scan(EdgeWeightedGraph G, int v)
{
marked[v] = true;
for (Edge e : G.adj(v))
{
int w = e.other(v);
if (marked[w]) // 边 v-w 两端都被遍历过,在队列中
continue;
if (e.weight() < distTo[w]) // 边 v-w 的权值小于顶点 w 的距离,说明加入该条边后生成树的总权值会下降
{
distTo[w] = e.weight(); // 加入边 v-w,更新 distTo 和 edgeTo
edgeTo[w] = e;
if (pq.contains(w)) // 搜若索队列中已经存在 w 则更新其 distTo(键值),不存在则将 w 加入
pq.decreaseKey(w, distTo[w]);
else
pq.insert(w, distTo[w]);
}
}
} public Iterable<Edge> edges()
{
Queue<Edge> mst = new Queue<Edge>();
for (int v = 0; v < edgeTo.length; v++)// 遍历边列表,把每个顶点对应的边加入队列中
{
Edge e = edgeTo[v];
if (e != null)
mst.enqueue(e);
}
return mst;
} public double weight()
{
double weight = 0.0;
for (Edge e : edges())
weight += e.weight();
return weight;
} public static void main(String[] args)
{
In in = new In(args[0]);
EdgeWeightedGraph G = new EdgeWeightedGraph(in);
class01 mst = new class01(G);
for (Edge e : mst.edges())
StdOut.println(e);
StdOut.printf("%.5f\n", mst.weight());
}
}

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

  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 10

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

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

    ▶ 书中第四章部分程序,包括在加上自己补充的代码,两种拓扑排序的方法 ● 拓扑排序 1 package package01; import edu.princeton.cs.algs4.Digraph ...

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

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

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

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

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

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

随机推荐

  1. 胖子哥的大数据之路(11)-我看Intel&&Cloudera的合作

    一.引言 5月8日,作为受邀嘉宾,参加了Intel与Cloudera在北京中国大饭店新闻发布会,两家公司宣布战略合作,该消息成为继Intel宣布放弃大数据平台之后的另外一个热点新闻.对于Intel的放 ...

  2. ALGO-140_蓝桥杯_算法训练_P1101

    有一份提货单,其数据项目有:商品名(MC).单价(DJ).数量(SL).定义一个结构体prut,其成员是上面的三项数据.在主函数中定义一个prut类型的结构体数组,输入每个元素的值,计算并输出提货单的 ...

  3. servlet.xml 出现 Referenced file contains errors(http://.......)

    问题描述: 打开Eclipse突然发现Web工程的servlet.xml突然报了红叉叉,错误信息如下: Referenced file contains errors (http://www.spri ...

  4. Razor语法记录

    虽然现在用着ASP.NET MVC但是cshtml使用Razor的标准形式去布局用的还是很少,这里就一点点把用到的关键点慢慢记下来,方便自己日后回忆吧! 1.将Action中返回的html字符串转换为 ...

  5. vc++获取网页源码之使用类型库(TypeLib)生成包装类

    1.在MFC项目名称上 右击->添加->选择Visual C++下的MFC->TypeLib中的MFC类->添加 可以从注册表表中共或是文件中根据相应的接口生成对应的包装类 效 ...

  6. kafka 生产消费原理详解

    Kafka日志及Topic数据清理 https://blog.csdn.net/qiaqia609/article/details/78899298 Kafka--Consumer消费者 pastin ...

  7. mysql 锁2

    官网地址 https://dev.mysql.com/doc/refman/5.5/en/innodb-transaction-isolation-levels.html 这里主要是说事务隔离级别,以 ...

  8. 几种常见NPE

    NPE(Null Point Exception的简称) 1.Map下的NPE 直接上代码: public class User { private Integer id; private Strin ...

  9. Win7删除缓存垃圾文件

    del /s /f /q C:\*.tmp

  10. windows server 2008通过任务计划程序定时访问网站

    1.新一个.bat文件,如: @echo offstart 网址exit 2.打windows server 2008,新建任务计划程序定时访问任务