【JAVA算法】图论算法 --求最小生成树Prim算法

写在前面:

    我也是一名java语言的爱好者,仅以此文作为学习的记录,对于文中出现的代码规范,代码格式,算法效率等问题,希望各路大神不吝赐教,在下感激不尽。同是学习的同学也同样希望互相交流,取长补短。                                                                                                                                            

——zsferrier@126.com

    


大体上来说,一个无向图G的最小生成树就是由该图的那些连接G的所有顶点的边构成的树,且其总价值最低

Prim算法

    计算最小生成树的一种方法是使其连续的一步步的长成,每一步都要把一个节点当作根并往上加边。在算法的任意时刻,我们都可以看到一组已经添加到树上的节点,而其余顶点尚未加到这棵树中。此时,算法在每一时刻都可以通过选择边(u,v)使得(u,v)的值是所有u在树上但v不在树上的边的值中的最小值而找出一个新的顶点添加到这棵树中。


算法设计:

    Prim算法基本和求最短路径的Dijsktra算法相同,设计数据类型时,对每个顶点保留dist和path以及一个known指标(表示顶点已知或未知)。这里dist是连接到已知顶点最短边的权,path是导致该顶点改变的最后顶点。

  数据类型:

	private static int MaxLen = 1000;
	private static Vertex[] verArray = new Vertex[7];
	class Vertex{
		public List<Integer> adjacency;
		public boolean known;
		public int dist;
		public Vertex path;
		public int vname;
		public Vertex(){
			known = false;
			dist = MaxLen;
			path = null;
		}
	}

算法伪代码

//Prim算法伪代码:
/**
 * Prim(Vertex s){
 * 		s.dist = 0;
 * 		while(there is an unknown vertex here){
 * 			Vertex v = the smallest dist vertex;
 * 			v.known = true;
 * 			for each Vertex w adjacent to v if( !w.known ) {
 * 				int dvw = the distance bewteen v and w;
 * 				if(dvw<w.dist){
 * 					w.dist = dvw;
 * 					w.path = v;
 * 				}
 * 			}
 * 		}
 * }
 * 
 */

Java语言实现Prim算法:

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

public class PrimA {

	/**
	 * @param args
	 */
	private static int MaxLen = 1000;
	private static Vertex[] verArray = new Vertex[7];
	class Vertex{
		public List<Integer> adjacency;
		public boolean known;
		public int dist;
		public Vertex path;
		public int vname;
		public Vertex(){
			known = false;
			dist = MaxLen;
			path = null;
		}
	}
	//判断是否还有unknown vertex存在
	public static boolean Judge(){
		for (Vertex v : verArray) {
			if (v.known==false) {
				return true;
			}
		}
		return false;
	}
	//返回Dist最小的unknown vertex
	public static Vertex SmallestVertex(){
		int Vnum = 0;
		int value = MaxLen;
		for(int i=0;i<7;i++){
			if((!verArray[i].known)&&(verArray[i].dist<value)){
				Vnum = i;
				value = verArray[i].dist;
			}
		}
		return verArray[Vnum];
	}
	public static void Prim(Vertex s){
		s.dist = 0;
		while(Judge()){
			Vertex v = SmallestVertex();
			v.known = true;
			for (Integer i : v.adjacency) {
				if ((!verArray[v.adjacency.indexOf(i)].known)&&(i<verArray[v.adjacency.indexOf(i)].dist)) {
					verArray[v.adjacency.indexOf(i)].dist = i;
					verArray[v.adjacency.indexOf(i)].path = v;
				}
			}
		}
	}
	public static void printTree(Vertex s){
		System.out.println("最小生成树为:");
		for (Vertex v : verArray) {
			
			if (v!=s) {
				System.out.println("V"+v.vname+"-----"+"V"+v.path.vname+"     "+"边权值为"+v.adjacency.get(v.path.vname));	
			}
		}
	}
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		//初始化
		verArray[0] = new PrimA().new Vertex();
		verArray[1] = new PrimA().new Vertex();
		verArray[2] = new PrimA().new Vertex();
		verArray[3] = new PrimA().new Vertex();
		verArray[4] = new PrimA().new Vertex();
		verArray[5] = new PrimA().new Vertex();
		verArray[6] = new PrimA().new Vertex();
		verArray[0].adjacency = new ArrayList<Integer>(Arrays.asList(0,2,4,1,MaxLen,MaxLen,MaxLen));
		verArray[0].vname = 0;
		verArray[1].adjacency = new ArrayList<Integer>(Arrays.asList(2,0,MaxLen,3,10,MaxLen,MaxLen));
		verArray[1].vname = 1;
		verArray[2].adjacency = new ArrayList<Integer>(Arrays.asList(4,MaxLen,0,2,MaxLen,5,MaxLen));
		verArray[2].vname = 2;
		verArray[3].adjacency = new ArrayList<Integer>(Arrays.asList(1,3,2,0,7,8,4));
		verArray[3].vname = 3;
		verArray[4].adjacency = new ArrayList<Integer>(Arrays.asList(MaxLen,10,MaxLen,7,0,MaxLen,6));
		verArray[4].vname = 4;
		verArray[5].adjacency = new ArrayList<Integer>(Arrays.asList(MaxLen,MaxLen,5,8,MaxLen,0,1));
		verArray[5].vname = 5;
		verArray[6].adjacency = new ArrayList<Integer>(Arrays.asList(MaxLen,MaxLen,MaxLen,4,6,1,0));
		verArray[6].vname = 6;
		//起点从v0开始
		Prim(verArray[0]);
		printTree(verArray[0]);
	}

}

运行结果:






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值