克鲁斯卡尔算法(公交站问题)
生活随笔
收集整理的這篇文章主要介紹了
克鲁斯卡尔算法(公交站问题)
小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
應(yīng)用場(chǎng)景
思路分析
代碼實(shí)現(xiàn)
package com.atguigu.kruskal;import java.util.Arrays;/*** @創(chuàng)建人 wdl* @創(chuàng)建時(shí)間 2021/4/6* @描述*/ public class KruskalCase {private int edgNum;//邊的個(gè)數(shù)private char[] vertexs;//頂點(diǎn)數(shù)組private int[][] matrix;//鄰接矩陣//使用INF表示兩個(gè)頂點(diǎn)不能連通private static final int INF = Integer.MAX_VALUE;public static void main(String[] args) {char[] vertexs = {'A', 'B', 'C', 'D', 'E', 'F', 'G'};//克魯斯卡爾算法的鄰接矩陣int matrix[][] = {/*A*//*B*//*C*//*D*//*E*//*F*//*G*//*A*/ {0, 12, INF, INF, INF, 16, 14},/*B*/ {12, 0, 10, INF, INF, 7, INF},/*C*/ {INF, 10, 0, 3, 5, 6, INF},/*D*/ {INF, INF, 3, 0, 4, INF, INF},/*E*/ {INF, INF, 5, 4, 0, 2, 8},/*F*/ {16, 7, 6, INF, 2, 0, 9},/*G*/ {14, INF, INF, INF, 8, 9, 0}};//大家可以在去測(cè)試其它的鄰接矩陣,結(jié)果都可以得到最小生成樹(shù).//創(chuàng)建KruskalCase對(duì)象實(shí)例KruskalCase kruskalCase = new KruskalCase(vertexs, matrix);//輸出構(gòu)建的kruskalCase.print();kruskalCase.kruskal();}//構(gòu)造器public KruskalCase(char[] vertexs, int[][] matrix) {//初始化頂點(diǎn)數(shù)和邊的個(gè)數(shù)int vlen = vertexs.length;//初始化頂點(diǎn),復(fù)制拷貝的方式this.vertexs = new char[vlen];for (int i = 0; i < vertexs.length; i++) {this.vertexs[i] = vertexs[i];} // this.vertexs=vertexs;//會(huì)修改原來(lái)的//初始化邊,使用的是復(fù)制拷貝的方式this.matrix = new int[vlen][vlen];for (int i = 0; i < vlen; i++) {for (int j = 0; j < vlen; j++) {this.matrix[i][j] = matrix[i][j];}}//統(tǒng)計(jì)邊的條數(shù)for (int i = 0; i < vlen; i++) {for (int j = i+1; j < vlen; j++) {if (this.matrix[i][j] != INF) {edgNum++;}}}}public void kruskal(){int index=0;//表示最后結(jié)果數(shù)組的索引int[] ends=new int[edgNum];//用于保存"已有最小生成樹(shù)"中的每個(gè)頂點(diǎn)在最小生成樹(shù)中的終點(diǎn)//創(chuàng)建結(jié)果數(shù)組,保存最后的最小生成樹(shù)EData[] rets=new EData[edgNum];//獲取圖中所有的邊的集合,一共有12條邊EData[] edges=getEdges();System.out.println("圖的邊的集合"+Arrays.toString(edges)+"共"+edges.length);//按照邊的權(quán)值大小進(jìn)行排序(從小到大)sortEdges(edges);//遍歷edges數(shù)組,將邊添加到最小生成樹(shù)中時(shí),判斷是否準(zhǔn)備加入的邊形成了回路,如果沒(méi)有,就加入rets,否則不能加入for (int i = 0; i < edgNum; i++) {//獲取到第i條邊的第一個(gè)頂點(diǎn)int p1 = getPosition(edges[i].start);//p1=4//獲取到第i條邊的第2個(gè)頂點(diǎn)int p2 = getPosition(edges[i].end);//p2=5//獲取p1這個(gè)頂點(diǎn)在已有的最小生成樹(shù)中的終點(diǎn)int m=getEnd(ends,p1);//獲取p2這個(gè)頂點(diǎn)在已有的最小生成樹(shù)中的終點(diǎn)int n=getEnd(ends,p2);//是否構(gòu)成回路if(m!=n){//沒(méi)有構(gòu)成回路ends[m]=n;//設(shè)置m在"已有最小生成樹(shù)"中的終點(diǎn) <E,F>[0,0,0,0,5,0,0,0,0,0,0,0]rets[index++]=edges[i];//有一條邊加入到rets數(shù)組}}//統(tǒng)計(jì)并打印最小生成樹(shù),輸入retsSystem.out.println("最小生成樹(shù)為");for (int i = 0; i < index; i++) {System.out.println(rets[i]);}}//打印鄰接矩陣public void print() {System.out.println("鄰接矩陣為:");for (int i = 0; i < vertexs.length; i++) {for (int j = 0; j < vertexs.length; j++) {System.out.printf("%10d\t", matrix[i][j]);}System.out.println();}}//對(duì)邊進(jìn)行排序處理,冒泡排序/*** @param edges 邊的集合*/private void sortEdges(EData[] edges) {for (int i = 0; i < edges.length-1; i++) {for (int j = 0; j < edges.length - 1 - i; j++) {if (edges[j].weight > edges[j + 1].weight) {EData temp = edges[j];edges[j] = edges[j+1];edges[j+1] = temp;}}}}/*** @param ch 頂點(diǎn)的值,比如'A','B'* @return 返回的是ch頂點(diǎn)對(duì)應(yīng)的下標(biāo),如果找不到,返回-1*/private int getPosition(char ch) {for (int i = 0; i < vertexs.length; i++) {if (vertexs[i] == ch) {//找到return i;}}//找不到,返回-1return -1;}/*** 獲取圖中的邊,放到EData[]中,后面我們需要遍歷該數(shù)組* 是通過(guò)matrix鄰接矩陣來(lái)獲取* EData 形式[['A','B',12],['B','F',7]...]** @return*/private EData[] getEdges() {int index = 0;EData[] edges = new EData[edgNum];for (int i = 0; i < vertexs.length; i++) {for (int j = i + 1; j < vertexs.length; j++) {if (matrix[i][j] != INF) {edges[index++] = new EData(vertexs[i], vertexs[j], matrix[i][j]);}}}return edges;}/*** 功能:獲取下標(biāo)為i的頂點(diǎn)的終點(diǎn),用于后面判斷兩個(gè)頂點(diǎn)的終點(diǎn)是否相同* @param ends:數(shù)組就是記錄了各個(gè)頂點(diǎn)對(duì)應(yīng)的終點(diǎn)是哪個(gè),ends數(shù)組是在遍歷過(guò)程中,逐步形成的* @param i 表示闖入的頂點(diǎn)對(duì)應(yīng)的下標(biāo)* @return 返回的就是下標(biāo)為i的這個(gè)頂點(diǎn)對(duì)應(yīng)的終點(diǎn)的下標(biāo)*/private int getEnd(int[] ends,int i){while (ends[i]!=0){i=ends[i];}return i;}}//創(chuàng)建一個(gè)類EData,它的對(duì)象實(shí)例就是表示一條邊 class EData {char start;//邊的一個(gè)點(diǎn)char end;//邊的另外一個(gè)點(diǎn)int weight;//邊的權(quán)值//構(gòu)造器public EData(char start, char end, int weight) {this.start = start;this.end = end;this.weight = weight;}//重寫toString方法,便于輸出邊的信息@Overridepublic String toString() {return "EData{" +"start=" + start +", end=" + end +", weight=" + weight +'}';} }總結(jié)
以上是生活随笔為你收集整理的克鲁斯卡尔算法(公交站问题)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 笔记本电脑算不算大功率电器?
- 下一篇: 2018蓝桥杯省赛---java---B