亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

java實現(xiàn)單源最短路徑

 更新時間:2019年01月21日 09:02:34   作者:浮生若夢yoo  
這篇文章主要為大家詳細介紹了java實現(xiàn)單源最短路徑,具有一定的參考價值,感興趣的小伙伴們可以參考一下

本文采用java實現(xiàn)單源最短路徑,并帶有略微詳細的注解,供大家參考,具體內容如下

package com.qf.greaph;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
import java.util.Map.Entry;

/**
 * @author jiayoo
 * 7 / 30 
 * Dijkstra最短路徑算法是一種單源最短路徑
 * 本文采用的是鄰接表表示圖。
 * 
 * 圖的表示: 1. 采用 ArrayList 來儲存 圖的頂點
 *   2. 采用 Map 來儲存 邊集 , map 可以 實現(xiàn) 一對多的關系, 因此能很好的實現(xiàn)鄰接表結構
 *   3. 采用ArrayList的原因 是使 邊集有序 這樣, Node 的里面 那個記錄距離的集合才能一一對應
 */

public class MinPath {

  private static class graph{
    private ArrayList<Node1> nodes = new ArrayList<>(); // 表示圖頂點 , 同時他也作為V集合
    private Map<Node1, ArrayList<Node1>> adjaNode = new HashMap<>(); // 表示圖的邊
    private ArrayList<Node1> nodes1 ; // 表示S集合, 即存儲已經訪問的節(jié)點,
    private float[] minPath; //用來存儲源點到每個頂點的距離
    float min = Float.MAX_VALUE;

    /**
     * @param start
     * @param end
     * @param distance
     * 構建鄰接表。使之成為圖
     */
    public void addAdjaNode(Node1 start, Node1 end, float distance) {

      if (!nodes.contains(start)) {
        nodes.add(start);
      }
      if (!nodes.contains(end)) {
        nodes.add(end);
      }
      if (adjaNode.containsKey(start) && adjaNode.get(start).contains(end)) {
        return ;
      }

      if (adjaNode.containsKey(start)) {
        adjaNode.get(start).add(end);
      }else {
        ArrayList<Node1> node = new ArrayList<Node1>();
        node.add(end);
        adjaNode.put(start, node);
      }
      start.distonext.add(distance);
    }

    /**
     * 將圖打印出來
     */
    public void prinGraph() {
      if (nodes == null || adjaNode == null) {
        System.out.println("圖為空");
        return ;
      }

      for (Entry<Node1, ArrayList<Node1>> entry : adjaNode.entrySet()) {
        System.out.println("頂點 : " + entry.getKey().name + " 鏈接頂點有: ");
        for(int i = 0; i < entry.getValue().size(); i++) {
          System.out.print(entry.getValue().get(i).name + " " + "距離是: " + entry.getKey().distonext.get(i) + ", ");
        }
        System.out.println();
      }
    }


    /**
     * 1.這個方法用于初始化S集合 及 初始化距離數(shù)組
     * 2. 設置源點, 并且將源點作為內容 初始化算法
     */
    public void findMinPath() {
      Node1 node1 = null; // 用來記錄列表里最小的點
      nodes1 = new ArrayList<>(); // 存儲已經遍歷過的點
      minPath = new float[nodes.size()]; // 初始化距離數(shù)組
      int i;
      /*
       * 對最短路徑進行初始化, 設置源點到其他地方的值為無窮大
       * */
      for (i = 0; i < minPath.length; i++) {
        minPath[i] = Float.MAX_VALUE;
      }
      Node1 node = nodes.get(0); 
      nodes1.add(node); // 將源點加入 S 集合
      node.visited = true;

      ArrayList<Node1> n = adjaNode.get(node); // 獲取到源點的邊集
      /*
       * 先對源節(jié)點進行初始化
       * 1. 對 距離數(shù)組進行初始化。
       * 2. 找到源點到某個距離最短的點, 并標記
       * 
       * */
      for (i = 0; i < n.size(); i++) {
        minPath[n.get(i).id] = node.distonext.get(i); // 最短路徑記錄
        if (min > node.distonext.get(i)) {
          min = node.distonext.get(i);
          node1 = n.get(i); // 找到當前最短路徑
        }
      }
      this.process(node1, min);
    }


    private void process(Node1 node, float distance ) {
      min = Float.MAX_VALUE; //作為標記
      Node1 node1 = null; // 同樣記錄距離最短的點
      int i;
      ArrayList<Node1> n = adjaNode.get(node); // 獲得邊集
      for (i = 0 ; i < n.size(); i++) {
        if (!n.get(i).visited) { // 這個邊集里的頂點不在 S 集合里
          if (minPath[n.get(i).id] == Float.MAX_VALUE) {
            minPath[n.get(i).id] = distance + node.distonext.get(i); // 源點到下一點的距離
          }else if (distance + node.distonext.get(i) < minPath[n.get(i).id] ) { //源點到該頂點的距離變小了, 則改變
            minPath[n.get(i).id] = distance + node.distonext.get(i); // 更新源點到下一個點的距離
          }
        }
      }
      /*
       * 這個for 用于找到 距離集合中 距離源點最近 且并未被訪問過的
       * 這個for 同時可以確保 該節(jié)點確實可到達
       * */

      for (i = 1; i < minPath.length; i++) {
        if (!nodes.get(i).visited) {
          if (min  > minPath[i] ) { 
            min = minPath[i];
            node1 = nodes.get(i);
          }
        }
      }
      if (node1 != null) {
        node1.visited = true;
        process(node1, min); //源點到 當前的距離
      }else { // 說明此位置沒有后續(xù)節(jié)點, 或者 已經全部被訪問完了, 則到達此位置只需要加上此位置的值

      }      
    }
  }

  public static void main(String[] args) {

    Node1 n1 = new Node1(0,"A");
    Node1 n2 = new Node1(1,"B");
    Node1 n3 = new Node1(2,"C");
    Node1 n4 = new Node1(3,"D");
    Node1 n5 = new Node1(4,"E");
    Node1 n6 = new Node1(5,"F");
    graph gp = new graph();
    gp.addAdjaNode(n1, n2, 6);
    gp.addAdjaNode(n2, n1, 6);
    gp.addAdjaNode(n1, n3, 3);
    gp.addAdjaNode(n3, n1, 3);


    gp.addAdjaNode(n2, n3, 2);
    gp.addAdjaNode(n3, n2, 2);
    gp.addAdjaNode(n2, n4, 5);
    gp.addAdjaNode(n4, n2, 5);

    gp.addAdjaNode(n3, n4, 3);
    gp.addAdjaNode(n4, n3, 3);
    gp.addAdjaNode(n3, n5, 4);
    gp.addAdjaNode(n5, n3, 4);

    gp.addAdjaNode(n4, n5, 2);
    gp.addAdjaNode(n5, n4, 2);
    gp.addAdjaNode(n4, n6, 3);
    gp.addAdjaNode(n6, n4, 3);

    gp.addAdjaNode(n5, n6, 5);
    gp.addAdjaNode(n6, n5, 5);

    // 下面嘗試一下非連通圖

//   /**
//    *   權值: 1
//    *  A -----------B
//    * 權 | *
//    * 值 | *  權值: 3
//    * 2  |  *
//    *   C-----D
//    * 權值: 5
//    * 
//    * 
//    * */
//   
//   gp.addAdjaNode(n1, n2, 1);
//   gp.addAdjaNode(n2, n1, 1);
//   
//   gp.addAdjaNode(n1, n3, 2);
//   gp.addAdjaNode(n3, n1, 2);
//   
//   gp.addAdjaNode(n1, n4, 3);
//   gp.addAdjaNode(n4, n1, 3);
//   
//   gp.addAdjaNode(n3, n4, 5);
//   gp.addAdjaNode(n4, n3, 5);
    gp.prinGraph();
    System.out.println("--------------------------------------------------------------------");
    System.out.println("此數(shù)組下標代表id,值代表從源點分別到各點的最短距離, A開始的下標是0, B、C、D等依次類推, 并且源點默認設置為id為零0的開始");
    gp.findMinPath();  
    System.out.println(Arrays.toString(gp.minPath));

  }

}


/**
 * 頂點類
 */
class Node1{
  String name; 
  boolean visited = false; // 訪問狀態(tài)。有效 減少原算法移除V集合中元素所花費的時間
  int id = -1; // 設置默認id為-1
  ArrayList<Float> distonext = new ArrayList<>(); //這一點 到另外每一個點的距離
  public Node1(int id, String name) {
    this.id = id;
    this.name = name;
  }

}


以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • Java農夫過河問題的繼承與多態(tài)實現(xiàn)詳解

    Java農夫過河問題的繼承與多態(tài)實現(xiàn)詳解

    這篇文章主要介紹了Java農夫過河問題的繼承與多態(tài)實現(xiàn)詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-01-01
  • SpringBoot發(fā)送短信驗證碼的實例

    SpringBoot發(fā)送短信驗證碼的實例

    第三方短信發(fā)送平臺有很多種,各個平臺有各自的優(yōu)缺點,在選擇的時候可以根據(jù)自己的具體實際情況定奪,本文主要介紹了SpringBoot發(fā)送短信驗證碼的實例,感興趣的可以了解一下
    2022-02-02
  • SpringCloud OpenFeign概述與使用

    SpringCloud OpenFeign概述與使用

    OpenFeign源于Netflix的Feign,是http通信的客戶端。屏蔽了網絡通信的細節(jié),直接面向接口的方式開發(fā),讓開發(fā)者感知不到網絡通信細節(jié)。所有遠程調用,都像調用本地方法一樣完成
    2023-01-01
  • Scala之文件讀取、寫入、控制臺操作的方法示例

    Scala之文件讀取、寫入、控制臺操作的方法示例

    這篇文章主要介紹了Scala之文件讀取、寫入、控制臺操作的方法示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-06-06
  • Springboot Mybatis-Plus數(shù)據(jù)庫單元測試實戰(zhàn)(三種方式)

    Springboot Mybatis-Plus數(shù)據(jù)庫單元測試實戰(zhàn)(三種方式)

    這篇文章主要介紹了Springboot Mybatis-Plus數(shù)據(jù)庫單元測試實戰(zhàn)(三種方式),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-12-12
  • SpringCloud @FeignClient參數(shù)的用法解析

    SpringCloud @FeignClient參數(shù)的用法解析

    這篇文章主要介紹了SpringCloud @FeignClient參數(shù)的用法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • springboot設置了server.port但是沒有用,還是8080問題

    springboot設置了server.port但是沒有用,還是8080問題

    這篇文章主要介紹了springboot設置了server.port但是沒有用,還是8080問題的解決,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • PowerJob的IdGenerateService工作流程源碼解讀

    PowerJob的IdGenerateService工作流程源碼解讀

    這篇文章主要為大家介紹了PowerJob的IdGenerateService工作流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2024-01-01
  • Java調用MySQL存儲過程并獲得返回值的方法

    Java調用MySQL存儲過程并獲得返回值的方法

    這篇文章主要介紹了Java調用MySQL存儲過程并獲得返回值的方法,實例分析了java實現(xiàn)MySQL存儲過程的相關技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-07-07
  • 說說字符串轉 OffSetDateTime 你真的會用嗎

    說說字符串轉 OffSetDateTime 你真的會用嗎

    這篇文章主要介紹了字符串轉 OffSetDateTime 你真的會用嗎?具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-08-08

最新評論