算法-数据结构(图)-弗洛伊德算法复现(Floyd)

弗洛伊德算法 (Floyd-Warshall算法)是一种用于求解所有节点对最短路径的动态规划算法,适用于有向图或无向图,且能处理带有负权边的图(但不能有负权环)。该算法的时间复杂度为 O(V3)O(V3),其中 VV 是图的节点数。


核心思想

弗洛伊德算法通过逐步优化路径来求解最短路径。其核心思想是:

  1. 对于任意两个节点 ii 和 jj,检查是否存在一个中间节点 kk,使得从 ii 到 jj 的路径可以通过 kk 变得更短。

  2. 通过动态规划逐步更新最短路径。


算法步骤

  1. 初始化

    • 创建一个距离矩阵 DD,其中 DijDij 表示节点 ii 到节点 jj 的最短路径长度。

    • 如果 ii 和 jj 之间有直接边,则 DijDij 为边的权重;否则为无穷大(∞∞)。

    • 对角线上的值 DiiDii 初始化为 0(节点到自身的距离为 0)。

  2. 动态规划更新

    • 对于每一个中间节点 kk(从 1 到 VV),更新所有节点对 ii 和 jj 的最短路径:

      Dij=min⁡(Dij,Dik+Dkj)Dij=min(Dij,Dik+Dkj)

    • 即,检查是否通过节点 kk 可以使 ii 到 jj 的路径更短。

  3. 结束

    • 当所有中间节点 kk 都被遍历后,矩阵 DD 中的值即为所有节点对的最短路径。

手动推导:

算法如下:

复制代码
public class Foloyd {
    //弗洛伊德最短路径算法复现
    public static void Floyd(int [] [] dist, int [][] path)
    {
        int L=dist.length;
        for(int k=0;k<L;k++)
        {
            for(int i=0;i<L;i++)
            {
                for(int j=0;j<L;j++)
                {
                    if(dist[i][k]!=Integer.MAX_VALUE&&dist[k][j]!=Integer.MAX_VALUE)
                    {
                        //判断是否通过中转路径能更短
                        if(dist[i][k]+dist[k][j]<dist[i][j])
                        {
                            //更新距离
                            dist[i][j]=dist[i][k]+dist[k][j];
                            //中转点
                            path[i][j]=k;
                        }
                    }
                }
            }
        }
        System.out.println("最短路径值为:");
        for(int i=0;i<L;i++)
        {
            for(int j=0;j<L;j++)
            {
                System.out.print(dist[i][j]);
                System.out.print(" ");
            }
            System.out.println();
        }
        System.out.println("最短路径为:");
        for(int i=0;i<L;i++)
        {
            for(int j=0;j<L;j++)
            {
                System.out.print(path[i][j]);
                System.out.print(" ");
            }
            System.out.println();
        }
    }

    public static void main(String[] args) {
        int[][] vG= {{0,6,13}, {10,0,4}, {5,Integer.MAX_VALUE,0}};
        int[][] path={{-1,-1,-1},{-1,-1,-1},{-1,-1,-1}};
        Floyd(vG,path);

    }
}

结果如下:

复制代码
最短路径值为:
0 6 10 
9 0 4 
5 11 0 
最短路径为:
-1 -1 1 
2 -1 -1 
-1 0 -1 
相关推荐
JieE21211 小时前
LeetCode 101. 对称二叉树|JS 递归 + 迭代双解法,彻底搞懂镜像判断
javascript·算法
nanxun88616 小时前
记一次诡异的 Docker 容器"串包"故障排查
java
用户15630681035118 小时前
Day01 | Java 基础(Java SE)
java
行者全栈架构师20 小时前
Maven dependency:tree 的 8 个高级用法
java·后端
行者全栈架构师1 天前
IDEA 中 Maven 项目的 15 个红色报错快速解决方法
java·后端
令人头秃的代码0_01 天前
mac(m5)平台编译openjdk
java
JieE2121 天前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack202 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树2 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
唐青枫2 天前
Java JDBC 实战指南:从 Connection 到事务和连接池
java