数据结构(12)Dijkstra算法JAVA版:图的最短路径问题

简介: 12.1.概述12.1.1.无权图的最短路径无权图的最短路径,即最少步数,使用BFS+贪心算法来求解最短路径,比较好实现,此处不做展开讨论。

12.1.概述

12.1.1.无权图的最短路径

无权图的最短路径,即最少步数,使用BFS+贪心算法来求解最短路径,比较好实现,此处不做展开讨论。

cb5502e3ea764783ad658c7759691e4f.png

12.1.2.带权图的最短路径

有权图的最短路径,不考虑权重为负数的情况,因为权重为负数的情况极有可能出现负值圈,在这个圈子上形成环路,最短路径是无限兜圈,趋于负无穷。

所以此处我们只考虑权重不为负数的带权图的最短路径求解问题。带权图的最短路径求解问题主要求两种最短路径:

  • 单源最短路径,某个点到全图各点之间的最短路径。
  • 多源最短路径,遍历全图的最短路径。

单源最短路径Dijkstra算法求解,多源最短路径用Floyd算法求解。

1.单源最短路径

单源最短路径用Dijkstra算法求解,Dijkstra算法其本质是个贪心算法。整个过程就是选择局部最优解,组成最后的解。

以下展示一个Dijkstra求解v1的单源最短路径的过程:

d3c666b361f74b538f71cafe30e5f7b4.png

记录两个值:

distance,到某个结点的最短距离,初始化值为无穷大,表示暂时未记录。

route,全局最短路径,初始化值为-1,表示暂时未记录。

下标 1 2 3 4 5 6 7
distance 无穷大 无穷大 无穷大 无穷大 无穷大 无穷大 无穷大
route -1 -1 -1 -1 -1 -1 -1


v1开始,刷新distance和route的值:

v1—>v1,distance为0

v1—>v2,distance为2<∞,将2刷新为v1—>v2的最短距离,将1(指代v1)刷新为最短路径。

v4同理……

下标 1 2 3 4 5 6 7
distance 0 2 无穷大 1 无穷大 无穷大 无穷大
route -1 1 -1 1 -1 -1 -1

扫描distance表发现distance最小的v4,于是将v4上得到的数据刷新进distance和route:

下标 1 2 3 4 5 6 7
distance 0 2 3 1 3 9 5
route -1 1 4 1 4 4 4

一直重复上面步骤,直到图里所有结点都被遍历一次,会得到如下结果:

下标 1 2 3 4 5 6 7
distance 0 2 3 1 3 6 5
route -1 1 4 1 4 7 4

distance记录的是v1到每个结点的最短路径,route里面记录的最大值是全局遍历一遍的最短路径。

2.多源最短路径

多源最短路径用floyd算法求解,多源最短路径不能只关注于当前最优解,还要关注全局最优解,求解此类问题一般使用动态规划,floyd算法就是个求解多源最短路径的经典动态规划算法。本文主要论述Dijkstra算法,floyd算法暂时不展开。

12.2.代码实现

以遍历如下无向图为例(为什么不遍历上面的例子喃?因为代码是很久前写的了。上面的例子是写文的时候新写的,偷个懒不想改代码了~嘿嘿):

b978f463464f43d9ad937d6112a796f2.png

public class Dijkstra {
    static int[][] graph;
    static int[] dist;
    static int[] path=new int[7];
    static boolean[] isUsed=new boolean[7];
    static {
        graph=new int[][]{
                {0,1,4,3,0,0,0},
                {1,0,3,0,0,0,0},
                {4,3,0,2,1,5,0},
                {3,0,2,0,2,0,0},
                {0,0,1,2,0,0,0},
                {0,0,5,0,0,0,2},
                {0,0,0,0,0,2,0}
        };
        dist=new int[]{Integer.MAX_VALUE,Integer.MAX_VALUE,Integer.MAX_VALUE,Integer.MAX_VALUE,Integer.MAX_VALUE,Integer.MAX_VALUE,Integer.MAX_VALUE};
    }
    public static void dijkstra(){
        while(true){
            //判断节点是否已经全部纳入
            if(isOver()){
                break;
            }
            //寻找未纳入的dist最小节点
            int i=findMin();
            //设置为已遍历状态
            isUsed[i]=true;
            //遍历该节点邻接节点
            for (int j=0;j<graph[i].length;j++) {
                if(graph[i][j]!=0&&isUsed[j]==false){
                    //更新dist、path
                    flashDistAndPath(i,j);
                }
            }
        }
    }
    public static int findMin(){
        int min=Integer.MAX_VALUE;
        int index=-1;
        for(int i=0;i<dist.length;i++){
            if(min>dist[i]&&isUsed[i]==false){
                min=dist[i];
                index=i;
            }
        }
        return index;
    }
    //之前的dist值一定是之前该节点到根节点的最短路径开销
    public static void flashDistAndPath(int i,int j){
            if(isUsed[j]==false) {
                if (graph[i][j] + dist[i] < dist[j]) {
                    dist[j] = graph[i][j] + dist[i];
                    path[j] = j;
                }
            }
    }
    public static boolean isOver(){
        int trues=0;
        for (boolean isused:isUsed) {
            if(isused==true){
                trues++;
            }
        }
        if(trues==dist.length){
            return true;
        }
        return false;
    }
    public static void main(String[] args) {
        isUsed[0]=true;
        dist[1]=1;
        dist[2]=4;
        dist[3]=3;
        path[1]=0;
        path[2]=0;
        path[3]=0;
        dijkstra();
        for (int i=0;i<dist.length;i++){
            System.out.println(dist[i]);
        }
    }
}


目录
相关文章
|
5天前
|
存储 安全 Java
Java并发编程中的高效数据结构:ConcurrentHashMap解析
【4月更文挑战第25天】在多线程环境下,高效的数据访问和管理是至关重要的。Java提供了多种并发集合来处理这种情境,其中ConcurrentHashMap是最广泛使用的一个。本文将深入分析ConcurrentHashMap的内部工作原理、性能特点以及它如何在保证线程安全的同时提供高并发性,最后将展示其在实际开发中的应用示例。
|
5天前
|
存储 算法
图的深度优先算法
图的深度优先算法
12 0
|
6天前
|
设计模式 算法 Java
[设计模式Java实现附plantuml源码~行为型]定义算法的框架——模板方法模式
[设计模式Java实现附plantuml源码~行为型]定义算法的框架——模板方法模式
|
7天前
|
算法 数据可视化 大数据
圆堆图circle packing算法可视化分析电商平台网红零食销量采集数据
圆堆图circle packing算法可视化分析电商平台网红零食销量采集数据
35 13
|
11天前
|
存储 供应链 Java
《Java 简易速速上手小册》第3章:Java 数据结构(2024 最新版)
《Java 简易速速上手小册》第3章:Java 数据结构(2024 最新版)
8 1
|
11天前
|
机器学习/深度学习 数据采集 算法
使用 Java 实现机器学习算法
【4月更文挑战第19天】Java在数据驱动时代为机器学习提供支持,具备丰富的数学和数据结构库,适用于实现线性回归、决策树、SVM和随机森林等算法。实现时注意数据预处理、模型选择、评估指标和可视化。利用Java的库和编程能力可构建高效模型,但需按问题需求选择合适技术和优化方法。
|
15天前
|
存储 机器学习/深度学习 算法
上机实验三 图的最小生成树算法设计 西安石油大学数据结构
上机实验三 图的最小生成树算法设计 西安石油大学数据结构
19 1
|
18天前
|
Java API
编码的奇迹:Java 21引入有序集合,数据结构再进化
编码的奇迹:Java 21引入有序集合,数据结构再进化
16 0
|
21天前
|
算法 安全 Java
java代码 实现AES_CMAC 算法测试
该代码实现了一个AES-CMAC算法的简单测试,使用Bouncy Castle作为安全提供者。静态变量K定义了固定密钥。`Aes_Cmac`函数接受密钥和消息,返回AES-CMAC生成的MAC值。在`main`方法中,程序对给定的消息进行AES-CMAC加密,然后模拟接收ECU的加密结果并进行比较。如果两者匹配,输出&quot;验证成功&quot;,否则输出&quot;验证失败&quot;。辅助方法包括将字节转为16进制字符串和将16进制字符串转为字节。
|
28天前
|
搜索推荐 Java
Java排序算法
Java排序算法
18 0