java 图的最小生成树问题 (代码自己写)

最小生成树是基于无无向图,并且是没有权值的图的。它的实现可以用深度遍历还有广度遍历实现

本代码是基于图的,所以和前面的    java 图 的那个代码是一样的 

不同是加入了  生成最小生成图的代码

 

 

老习惯理论:

Java代码   收藏代码
  1. /** 
  2.  *  
  3. 详细代码后面会给出来的,现在只是大概的只是代码怎么写,怎么个思路 
  4. package endual; 
  5.  
  6. public class ShortTree { 
  7.  
  8.     //mst 
  9.     public void mst() { 
  10.          
  11.      
  12.         while (!(theStack).isEmpty()) { 
  13.              
  14.             int currentVertex = theStack.peek() ; 
  15.             //get next univisited heighbor 
  16.             //找到下一个邻居的未访问的顶点 
  17.              
  18.             int v = getAdUnvisitedVertex(currentVertex) ; 
  19.             if (v == -1) {//如果不存在,那么跳出一个栈里面的作为当前的点 
  20.                 theStack.pop() ;  
  21.             } 
  22.             else { //否则就是要 
  23.                  
  24.                 vertexList[v].isVisited = true ; //已经被访问过了的。然后就是true了 
  25.                 theStack.push(v) ; //压入栈  
  26.                  
  27.                 displayVertex(currentVertex) ; //打印出访问过的点 
  28.                 displayVertex(v) ; 
  29.                 System.out.println(" * * "); 
  30.                  
  31.                  
  32.             } 
  33.              
  34.         } 
  35.          
  36.     } 
  37.      
  38.      
  39.     for(int i=0; i < nVertex; i++) { 
  40.          
  41.         vertext[j].isVisited() = false ; 
  42.     } 
  43.      
  44. } 
  45.  */  

 

 

 

 

最小生成树其实找的是连接顶点的用最少的边,并不用去考虑边的长度,这个不是带权的边,所以是不用考虑的。

需要注意的是

    最小生成树的边数E总是比该图的所以顶点之和少一个

  E = V -1 ;

 

 记住,不必关心边的长度,并不需要找到一条最短路径,而是要找最少数量的边。

 

 创建最小生成树的算法与搜索的算法几乎是相同的。它同样可以基于广度搜索或者是深度搜索算法来实现

 

 我们用深度搜索算法来实现。

 

Java代码   收藏代码
  1. import java.util.LinkedList;  
  2. import java.util.Queue;  
  3. import java.util.Stack;  
  4. import java.util.concurrent.ArrayBlockingQueue;  
  5.   
  6. import javax.management.Query;  
  7.   
  8. /** 
  9.  * 下面来看下Graph类,包含的是创建连接链表和连接矩阵的方法,以向Graph对象插入顶点和边的方法 
  10.  * (最小生产树的方法的代码在最后面,代码请看 前面的图那个文章  ,这只是一部分代码) 
  11.  * @author Endual 
  12.  * 
  13.  */  
  14. public class Graph {  
  15.   
  16.     private  final int MAX_VERTS = 20 ;  
  17.     private  Vertex[] vertexList ; //存放顶点的数组  
  18.     private  int adjMat[][] ; //表示用矩阵的形式来表示边有没有  
  19.     private  int nVerts ; //当前的图中有多少个顶点了  
  20.       
  21.     public Graph() {  
  22.         super();  
  23.         this.vertexList = new Vertex[this.MAX_VERTS];  
  24.         this.nVerts = 0 ;  
  25.     this.initialAdjMat () ; //初始化矩阵的  
  26.     }  
  27.     //初始化矩阵的  
  28.     private void initialAdjMat() {  
  29.           
  30.         for (int i=0; i < this.MAX_VERTS; i++) {  
  31.             for (int j=0; j < this.MAX_VERTS; j++) {  
  32.                 this.adjMat[i][j] = 0 ; //初始数组  
  33.             }  
  34.         }  
  35.           
  36.     }  
  37.     //  
  38.     //创建图 = 创建顶点和创建边,两个步骤是分开的  
  39.     //添加顶点  
  40.     public void addVertex(char lab) {  
  41.           
  42.         Vertex ver = new Vertex(lab) ;  
  43.         this.vertexList[this.nVerts] = ver ;  
  44.         this.nVerts++ ;  
  45.           
  46.     }  
  47.       
  48.     //创建边(start 和 end 要有范围的 从0 到 N-1)  
  49.     public void addEdge(int start, int end) {  
  50.           
  51.         this.adjMat[start][end] = 1 ;  
  52.         this.adjMat[end][start] = 1 ;  
  53.           
  54.     }  
  55.    ///  
  56.     //显示一个顶点  
  57.     public void displayVertex(int v) {  
  58.           
  59.         System.out.println(this.vertexList[v].getLabel()) ;  
  60.           
  61.     }  
  62.     //  
  63.     /** 
  64.      * 搜索 
  65.      * 在图中实现的最基本的操作之一就是搜索从一个指定顶点可以到达哪些顶点可以到达哪些顶点。例如,可以想象要找出美国有多少个城市 
  66.      * 可以从kansan乘坐旅行车到达(当然了中间是不能换车的)。一些城市可以直达,而有些城市因为没有旅行列车服务而不能到达。 
  67.      * 有些地方即使有列子服务也不能到达,因为他们的铁轨不能和出发或者沿途的标准铁轨系统相连。 
  68.      *  
  69.      * 还有一种情况就是需要找出所有的当前顶点可以到达的顶点。 
  70.      * 假设有这么一张图,现在需要一种算法来提供系统的方法,从某个特定顶点开始,沿着边移动到其他顶点,移动完毕后,需要保证访问了喝 
  71.      * 起始点相连的每一个顶点。 
  72.      * 有两种常用的方法可以用来搜索图;深度优先搜索和广度优先搜索,他们最终都会到达所有连通的顶点 
  73.      * 
  74.      * 深度优先搜索通过栈来实现,而广度优先搜索通过队列实现。 
  75.      *  
  76.      *  
  77.      * 深度优先搜索 
  78.      *  
  79.      * 在搜索到尽头的时候,深度优先搜索用栈记住下一步的走向,这里展示了一个例子, 
  80.      * 为了实现深度优先搜索,找到一个起始本例的顶点是A 
  81.      * 需要做的三件事情: 
  82.      * 首先是访问该顶点,然后是把该节点放入到栈中去,以便记住它 
  83.      * 最后标记该节点,这样就不会再访问它了 
  84.      * 下面就可以访问任何与顶点A相连的顶点了,只要还没有访问过它,假设顶点按字母顺序访问,所以下面访问顶点B。然后标记它,放入栈中。 
  85.      *  已经访问过B,做相同的事情,找下一个为访问的节点的顶点,也就是F,这个过程就叫做规则1。 
  86.      *   
  87.      *  规则1: 
  88.      *       如果可能,访问一个邻接的未访问顶点,标记它,并把它放入到栈中。 
  89.      *    
  90.      *   再次应用规则1,这次访问顶点H。然而,这里还需要做一些事情,因为没有和H邻接的未访问顶点。 
  91.      *    
  92.      *   规则2. 
  93.      *      当不能执行规则1的时候,如果栈不空,就从栈中探出一个顶点 
  94.      *       
  95.      *       根据这条规则,从栈中探出H,这样就又回到了顶点F,F也没有和邻接且邻接没有访问的顶点了。那么再弹出F,这回到顶点B。这时 
  96.      *  只有顶点A在栈中了。 
  97.      *       然而A还没有访问过的连接点,所以访问下一个顶点C,但是C又是在这一条线路的终点,所以栈中弹出它,再次回到A点。接着访问D,G和I。 
  98.      *  当到达I时,把它们都弹出栈,现在回到A然后访问E,最后再次回到A 
  99.      *      
  100.      *   规则3 
  101.      *      如果不能执行规则1 和规则 2,那么就完成了整一个深度搜索的过程了 
  102.      *  栈的内容就是刚才起始顶点到各个顶点访问的整个过程。从起始顶点出发访问下一个顶点时,就是把这个顶点入栈,回到起始顶点,出栈 
  103.      *  访问顶点的顺序是ABFHCDGIE 
  104.      *   
  105.      *    深度优先搜索算法要得到距离起始点最远的顶点,然后再不能继续前进的时候返回,使用深度这个属于表示与起始点的距离,便可以理解深度优先 
  106.      *    搜索的意义。 
  107.      *     
  108.      *    模拟问题 
  109.      *     
  110.      *    深度优先搜索与迷宫问题有类似的,迷宫在英国很流行,可以由一方给另一方设置障碍,由另一方想办法通过,迷宫由狭小的过道和过道的交汇点组成 
  111.      *     
  112.      *                                                                                                                                              
  113.      *      JAVA代码 深度优先搜索的关键在于能够找到与某一顶点邻接且没有访问过的顶点。邻接矩阵是关键,找到指定顶点所在的行。第一列开始向后寻找值为1的列; 
  114.      *      列号是邻接顶点的号码。检验这个顶点是不是被访问过的,如果是这样的,那么就是要访问的下个顶点,如果该行没有顶点既等于1,且又是没有访问过的,那么 
  115.      *      与指定点相邻接的顶点就全部访问过了。这个过程在下面的代码中实现                                                                                                                                    
  116.      *                                                                                                                                              
  117.      *  
  118.      *  
  119.      */  
  120.       
  121.       
  122.       
  123.     public int getAdjUnivisitedVertex(int v) { //从哪个顶点开始访问  
  124.           
  125.         for(int j=0; j < this.nVerts; j++) {  
  126.             if (this.adjMat[v][j] == 1 && this.vertexList[j].isVisisted == false) {  
  127.                 return j ;  
  128.             }  
  129.               
  130.         }  
  131.           
  132.         return -1 ;//如果没有访问到,那么返回-1 ;  
  133.     }  
  134.       
  135.     //考察dfs()方法,这个方法实际执行了深度的优先搜索。包含三条规则。它循环执行,知道栈为空,每次循环中,有四件事情  
  136.     /** 
  137.      * 1. 用peek()方法检查栈顶的顶点 
  138.      * 2. 试图找到这个顶点还未访问的邻接点 
  139.      * 3. 如果找没有找到,出栈 
  140.      * 4. 如果找到了这样的顶点,访问这个顶点,并且把它压入栈 
  141.      */  
  142.     //深度搜索  
  143.       
  144.     public void dfs() {  
  145.           
  146.         Stack theStack = new Stack() ;  
  147.         //所谓遍历是从第一个顶点开始的 也是 vertesList[0]开始的  
  148.         this.vertexList[0].isVisisted = true ;  
  149.         this.displayVertex(0) ;  
  150.         theStack.push(0) ;  
  151.           
  152.         while(!theStack.isEmpty()) {  
  153.             int v = this.getAdjUnivisitedVertex((Integer) theStack.peek()) ;  
  154.             if (-1 == v) { //已经被访问过的  
  155.                 theStack.pop() ;//弹出一个  
  156.                   
  157.             }  
  158.             else {  
  159.                 this.vertexList[v].isVisisted = true ;  
  160.                 this.displayVertex(v) ;  
  161.                 theStack.push(v) ;  
  162.             }  
  163.               
  164.         }  
  165.       
  166.         //stack 是空的话,就做  
  167.         for(int i=0; i < this.nVerts; i++) {  
  168.               
  169.             this.vertexList[i].isVisisted = false ; //重新初始化节点,为下次的使用  
  170.         }  
  171.           
  172.     } //end dfs  
  173.       
  174.     /** 
  175.      * 广度优先搜索 
  176.      *  
  177.      *   正如深度优先搜索中看到的,算法表现得好像要尽快地远离起始点的。相反,在广度优先搜索中,算法好像要尽可能地靠近起始点 
  178.      *   它首先访问的是顶点的所有邻接点,然后再访问较远的区域。这种搜索不能用栈,只能用到队列来实现 
  179.      *    
  180.      *   一个例子 
  181.      *    
  182.      *   A是起始点,所有访问它,并且标记为当前空的顶点。然后应用下面几条规则 
  183.      *    
  184.      *   规则1 
  185.      *      访问下一个未来访问的邻接点(如果存在), 这个顶点必须是当前顶点的邻接点,标记它,并把它插入到队列中 
  186.      *      (当前节点为改变) 
  187.      *   规则2 
  188.      *      如果因为已经没有未访问顶点而不能执行规则1, 那么从队列头取一个顶点(如果有),并且使得成为当前的顶点。 
  189.      *      (当前节点改变了) 
  190.      *   规则3 
  191.      *      如果因为队列为空而不能执行规则2.那么广度优先搜索的到此就结束了 
  192.      *       
  193.      *       
  194.      *      A-------B--------C------D 
  195.      *       \       |      X 
  196.      *        \      |      | 
  197.      *         \-----e------F----G 
  198.      *                     |   
  199.                            Y----H 
  200.       
  201.         
  202.           在每一个时刻队列中包含的顶点是那些本身已经被访问,而它的邻居还未被访问的顶点。(对比深度优先搜索,栈的内容是起始点 
  203.           到当前顶点经过的所有顶点)顶点访问的顺序。 
  204.            
  205.          深度优先遍历和广度优先遍历的异同点  
  206.            可以认为广度优先搜索就像往水中投入石块,水波纹扩展的过程===对于喜爱流行病学的人来说,就好比流感通过航空旅行客从一个城市 
  207.            到另一个城市了。首先是相距起始点只有一条边的所有顶点被访问,然后是相距两条边的所有顶点被访问,一次类推 
  208.             
  209.      */  
  210.       
  211.       
  212.       
  213.     //广度搜索与深度类似的,只是用队列代替了栈,嵌套的循环代替了单层循环。外层循环等待队列为空,而内层循环依次寻找当前顶点的未访问邻接点  
  214.     //下面是代码  
  215.       
  216.     public  void bfs() {  
  217.           
  218.         Queue theQueue = new LinkedList();    
  219.           
  220.         this.vertexList[0].isVisisted = true ;  
  221.         this.displayVertex(0) ;  
  222.         theQueue.add(0) ;  
  223.           
  224.         int v2 ;  
  225.           
  226.         while (!theQueue.isEmpty()) {  
  227.               
  228.             int v1 = (Integer) theQueue.remove() ;  
  229.             v2 = this.getAdjUnivisitedVertex(v1) ;  
  230.             while (v2 != -1) {  
  231.                   
  232.                 this.vertexList[v2].isVisisted = true ;  
  233.                 this.displayVertex(v2) ;  
  234.                 theQueue.add(v2) ;  
  235.             }  
  236.               
  237.         }  
  238.           
  239.         for(int i=0; i < this.nVerts; i++) {  
  240.               
  241.             this.vertexList[i].isVisisted = false ; //重新初始化节点,为下次的使用  
  242.         }  
  243.     }  
  244.       
  245.     /* 
  246.      *  
  247.      *  
  248.      * 请看我请看我请看我 
  249.      *  
  250.      *  
  251.      *  
  252.      *  
  253.      */  
  254.       
  255.       
  256.     //最小生成树的代码  
  257.     public void mst() {  
  258.         Stack theStack = new Stack() ;  
  259.         this.vertexList[0].isVisisted = true ; //从顶点开是访问,所以立马设置ture  
  260.         //得到它的邻接的  
  261.           
  262.         theStack.push(0) ; //从节点0开始访问,压入到栈中  
  263.           
  264.         while (!theStack.isEmpty()) { //当前的栈不为空  
  265.               
  266.           int currentVertex = (Integer) theStack.peek() ; //只是看,没删删除掉最上面的那个元素  
  267.           //取得当前顶点的邻接顶点  
  268.           int v = this.getAdjUnivisitedVertex(currentVertex) ;            
  269.             if(v == -1) {  
  270.                   
  271.                 theStack.pop() ;  
  272.             } //如果没有找到当前顶点的邻接的顶点,那么弹出一个顶点  
  273.             else {  
  274.                   
  275.                 //否则就是访问  
  276.                 this.vertexList[v].isVisisted = true ;  
  277.                 theStack.push(v) ;  
  278.                   
  279.                 this.displayVertex(currentVertex) ; //打印当前的节点  
  280.                 this.displayVertex(v) ; //大于邻接节点  
  281.                 System.out.println(" ") ;  
  282.                   
  283.             } //end 当这个栈的空间都没有的时候,那么就停止操作  
  284.         }  
  285.           
  286.         //初始化图的顶点未访问过的,下次用的时候还是还可以访问  
  287.           
  288.         for(int i=0; i < this.nVerts; i++) {  
  289.             this.vertexList[i].isVisisted = false ; //全部设置成没有访问过的  
  290.         }  
  291.               
  292.           
  293.           
  294.     }  
  295.       
  296.       
  297.       
  298.       
  299.       
  300.       
  301.       
  302.       
  303.       
  304.       
  305.       
  306.       
  307.       
  308.       
  309.       
  310.       
  311.   
  312. }  
下面是使用 Java 实现 Prim 算法的示例代码: ```java import java.util.Arrays; public class PrimMST { private static final int V = 5; // 中顶点的数量 // 查找最小权值的顶点 private int minKey(int[] key, boolean[] mstSet) { int min = Integer.MAX_VALUE; int minIndex = -1; for (int i = 0; i < V; i++) { if (!mstSet[i] && key[i] < min) { min = key[i]; minIndex = i; } } return minIndex; } // 打印生成的 MST private void printMST(int[] parent, int[][] graph) { System.out.println("Edge \tWeight"); for (int i = 1; i < V; i++) { System.out.println(parent[i] + " - " + i + "\t" + graph[i][parent[i]]); } } // 使用 Prim 算法查找 MST private void primMST(int[][] graph) { int[] parent = new int[V]; // 存储 MST int[] key = new int[V]; // 存储每个顶点的键值 boolean[] mstSet = new boolean[V]; // 判断顶点是否已经在 MST 中 // 初始化 key 为最大值,mstSet 为 false Arrays.fill(key, Integer.MAX_VALUE); Arrays.fill(mstSet, false); // 将第一个顶点加入 MST key[0] = 0; parent[0] = -1; // 一共需要添加 V-1 个顶点到 MST 中 for (int i = 0; i < V - 1; i++) { // 选择最小键值的顶点 int u = minKey(key, mstSet); // 将该顶点加入 MST mstSet[u] = true; // 更新相邻顶点的键值 for (int v = 0; v < V; v++) { if (graph[u][v] != 0 && !mstSet[v] && graph[u][v] < key[v]) { parent[v] = u; key[v] = graph[u][v]; } } } // 打印生成的 MST printMST(parent, graph); } public static void main(String[] args) { PrimMST prim = new PrimMST(); int[][] graph = new int[][]{{0, 2, 0, 6, 0}, {2, 0, 3, 8, 5}, {0, 3, 0, 0, 7}, {6, 8, 0, 0, 9}, {0, 5, 7, 9, 0}}; prim.primMST(graph); } } ``` 该代码使用邻接矩阵表示,示例的邻接矩阵为: ``` 2 3 6 +--+---+--+---+--+ | 0| 2 | 0| 6 | 0| +--+---+--+---+--+ | 2| 0 | 3| 8 | 5| +--+---+--+---+--+ | 0| 3 | 0| 0 | 7| +--+---+--+---+--+ | 6| 8 | 0| 0 | 9| +--+---+--+---+--+ | 0| 5 | 7| 9 | 0| +--+---+--+---+--+ ``` 输出结果为: ``` Edge Weight 0 - 1 2 1 - 2 3 0 - 3 6 1 - 4 5 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值