迪杰斯特拉算法求图的最短路径(java)

news2025/1/18 5:32:24

迪杰斯特拉算法

  • 图的最短路径的解法

单源最短路径

  • 从一个点开始,可以找到其中任意一个点的最短路径。

多源最短路径

  • 从任何一个点开始,可以找到其中任何一个点的最短路径。

解题过程

给定一个带权有向图G=(G, V), 另外,还给定 V 中的一个顶点,称为源。现在我们要计算从

源到所有其他各顶点的最短路径长度。这里的长度是指路上各边权之和。这个问题通常称为单

源最短路径问题。

Dijkstra算法的解决方案

即先求出长度最短的一条最短路径,再参照它求出长度次短的一条最短路径,依次类推,直到

在这里插入图片描述

1.选出一个顶点v作为源点,并将从源点v出发的所有边到各顶点的最短路径

创建一个数组dist,记录从这个点到其他点之间的距离

没有直接相连,就把两点之间的距离设置为无穷大

创建vistited状态数组,记录节点是否被访问

int index = startIndex;// 当前节点到自身的距离为0
this.dist[index] = 0;// 源点已被访问
visited[index] = true;// 3.从源到各个节点的最短距离LinkedList<AbstractMap.SimpleEntry<Integer, Integer>> list = this.adj[index];
for (int i = 0; i < list.size(); i++) {    
    AbstractMap.SimpleEntry<Integer, Integer> entry = list.get(i);    		      this.dist[entry.getKey()] = entry.getValue();
}

2.设置源点到自身的距离为0,并设置与源点直接相连点的距离

在这里插入图片描述

3.遍历距离表,循环的次数等于层数

​ 在距离表中,从未被访问的节点中找一个距离最小的节点

将该节点设置为已访问,并将于该节点邻接的节点的值分别进行求和与其邻接的距离表中的值进行比较,如果小于距离表中的值,则进行替换

        for (int i = 0; i < this.dist.length; i++) { // 循环的次数=层数
            // 4.在距离表中,从未被访问的节点中找一个距离最小的节点
            int minDistance = Integer.MAX_VALUE;
            int minDistanceIndex = -1;
            for (int j = 0; j < this.dist.length; j++) {
                if (!this.visited[j] && this.dist[j] < minDistance) {
                    minDistance = this.dist[j];
                    minDistanceIndex = j;
                }
                /*if (!this.visited[j] && this.dist[j] != Integer.MAX_VALUE) {
                    minDistance = Math.min(minDistance, this.dist[j]);
                    minDistanceIndex = j;
                }*/
            }
            // 特殊处理:源与任何节点都不连接
            if (minDistance == Integer.MAX_VALUE) {
                break;
            }

            // 调整dist中各项点到源点的路径
            // 先设置minDistanceIndex 已经被访问过
            this.visited[minDistanceIndex] = true;
            // 获取minDistanceIndex的邻接表
            LinkedList<AbstractMap.SimpleEntry<Integer, Integer>> list2 = this.adj[minDistanceIndex];
            // 使用 minDistance+curNode的权值 与 距离表中的原来的值进行比较,如果小的话就更新
            for (int k = 0; k < list2.size(); k++) {
                AbstractMap.SimpleEntry<Integer, Integer> entry2 = list2.get(k);
                int key = entry2.getKey();
                int value = entry2.getValue();
                if(this.visited[key]){
                    continue;
                }
                if (minDistance + value < this.dist[key]) {
                    this.dist[key] = minDistance + value;
                }
            }
        }

4.重复步骤3,直到任何节点都被访问

所有代码

package graph;

import java.util.AbstractMap;
import java.util.Arrays;
import java.util.LinkedList;

/**
 * 迪杰斯特拉算法
 */
public class Graph03 {

    // 邻接表,存储的是每一个节点的邻接节点与邻接节点的“关系”(key是索引,value是权值)
    private LinkedList<AbstractMap.SimpleEntry<Integer, Integer>>[] adj; // 邻接表

    private Vertex[] vertexes; // 节点集合


    // 1.创建一个距离表dist,存放从源到其他节点的最短距离
    private int[] dist;

    // 创建状态数组,记录节点是否被访问
    private boolean[] visited;

    // 图的构造
    public Graph03(int size, String[] arr) {
        this.vertexes = new Vertex[size];
        // 初始化邻接表,创建每个节点list
        this.adj = new LinkedList[size];
        for (int i = 0; i < size; i++) {
            this.vertexes[i] = new Vertex(arr[i], i);
            // 每个节点对应的list不能为空,将每一个节点的list实体化
            this.adj[i] = new LinkedList<>();
        }
        // 2.对距离表进行初始化
        dist = new int[size];
        Arrays.fill(dist, Integer.MAX_VALUE);
        visited = new boolean[size];
    }

    /**
     * @param startIndex 代表源节点的索引
     */
    public int[] dijkstra(int startIndex) {
        int index = startIndex;
        // 当前节点到自身的距离为0
        this.dist[index] = 0;
        // 源点已被访问
        visited[index] = true;
        // 3.从源到各个节点的最短距离
        LinkedList<AbstractMap.SimpleEntry<Integer, Integer>> list = this.adj[index];
        for (int i = 0; i < list.size(); i++) {
            AbstractMap.SimpleEntry<Integer, Integer> entry = list.get(i);
            this.dist[entry.getKey()] = entry.getValue();
        }

        for (int i = 0; i < this.dist.length; i++) { // 循环的次数=层数
            // 4.在距离表中,从未被访问的节点中找一个距离最小的节点
            int minDistance = Integer.MAX_VALUE;
            int minDistanceIndex = -1;
            for (int j = 0; j < this.dist.length; j++) {
                if (!this.visited[j] && this.dist[j] < minDistance) {
                    minDistance = this.dist[j];
                    minDistanceIndex = j;
                }
                /*if (!this.visited[j] && this.dist[j] != Integer.MAX_VALUE) {
                    minDistance = Math.min(minDistance, this.dist[j]);
                    minDistanceIndex = j;
                }*/
            }
            // 特殊处理:源与任何节点都不连接
            if (minDistance == Integer.MAX_VALUE) {
                break;
            }

            // 调整dist中各项点到源点的路径
            // 先设置minDistanceIndex 已经被访问过
            this.visited[minDistanceIndex] = true;
            // 获取minDistanceIndex的邻接表
            LinkedList<AbstractMap.SimpleEntry<Integer, Integer>> list2 = this.adj[minDistanceIndex];
            // 使用 minDistance+curNode的权值 与 距离表中的原来的值进行比较,如果小的话就更新
            for (int k = 0; k < list2.size(); k++) {
                AbstractMap.SimpleEntry<Integer, Integer> entry2 = list2.get(k);
                int key = entry2.getKey();
                int value = entry2.getValue();
                if(this.visited[key]){
                    continue;
                }
                if (minDistance + value < this.dist[key]) {
                    this.dist[key] = minDistance + value;
                }
            }
        }
        return this.dist;
    }

    // 节点的类
    class Vertex {

        private String value; // 结点内容

        private int index; // 结点在数组中的索引

        private int weight; // 权

        Vertex(String value, int index, int weight) {
            this.value = value;
            this.index = index;
            this.weight = weight;
        }

        Vertex(String value, int index) {
            this(value, index, 1);
        }
    }

    public void print() {
        System.out.println("---邻接表---");
        for (int i = 0; i < this.adj.length; i++) {
            System.out.println();
            for (int j = 0; j < this.adj[i].size(); j++) {
                System.out.print(this.adj[i].get(j).getKey() + "----->");
            }
        }
        System.out.println();
    }

    public static void main(String[] args) {

        String[] arr = {"A", "E", "F", "G", "H", "P"};
        int size = arr.length;
        Graph03 graph02 = new Graph03(size, arr);
        graph02.adj[0].add(new AbstractMap.SimpleEntry<>(1, 5));
        graph02.adj[0].add(new AbstractMap.SimpleEntry<>(2, 4));
        graph02.adj[0].add(new AbstractMap.SimpleEntry<>(3, 2));

        graph02.adj[1].add(new AbstractMap.SimpleEntry<>(0, 5));
        graph02.adj[1].add(new AbstractMap.SimpleEntry<>(3, 1));
        graph02.adj[1].add(new AbstractMap.SimpleEntry<>(4, 3));

        graph02.adj[2].add(new AbstractMap.SimpleEntry<>(0, 4));

        graph02.adj[3].add(new AbstractMap.SimpleEntry<>(0, 2));
        graph02.adj[3].add(new AbstractMap.SimpleEntry<>(1, 1));
        graph02.adj[3].add(new AbstractMap.SimpleEntry<>(4, 2));
        graph02.adj[3].add(new AbstractMap.SimpleEntry<>(5, 4));

        graph02.adj[4].add(new AbstractMap.SimpleEntry<>(1, 3));
        graph02.adj[4].add(new AbstractMap.SimpleEntry<>(3, 2));
        graph02.adj[4].add(new AbstractMap.SimpleEntry<>(5, 3));

        graph02.adj[5].add(new AbstractMap.SimpleEntry<>(3, 4));
        graph02.adj[5].add(new AbstractMap.SimpleEntry<>(4, 3));

        graph02.print();

        int[] dist = graph02.dijkstra(0);
        for(Integer i:dist){
            System.out.println(i);
        }


    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/85108.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

力扣(LeetCode)1832. 判断句子是否为全字母句(C++)

哈希集合1 哈希集合记录 262626 个字母是否出现&#xff0c;一次遍历字符串&#xff0c;维护哈希集合&#xff0c;同时维护答案。遍历完成&#xff0c;仅当答案等于 262626 &#xff0c;句子是全字母句。 class Solution { public:bool checkIfPangram(string sentence) {boo…

轻松提高性能和并发度,springboot简单几步集成缓存

目录 1、缘由 2、技术介绍 2.1、技术调研 2.2、spring支持的cache 2.3、cache的核心注解 2.3.1 EnableCaching 2.3.2 Cacheable 2.3.3 CachePut 2.3.4 CacheEvict 2.4 cache的架构 2.5 cachemanager的实现类 3、搞个例子 3.1 为什么使用redis 作为缓存 3.2 代码走起…

【虚幻引擎】UE4/UE5数字孪生与前端Web页面匹配

一、数字孪生 数字孪生是一种多维动态的数字映射&#xff0c;可大幅提高效能。数字孪生是充分利用物理模型、传感器更新、运行历史等数据&#xff0c;集成多学科、多物理量、多尺度、多概率的仿真过程&#xff0c;在虚拟空间中完成对现实体的复制和映射&#xff0c;从而反映物理…

MySQL常用窗口函数

1、窗口函数概念 窗口的概念非常重要&#xff0c;它可以理解为记录集合&#xff0c;窗口函数也就是在满足某种条件的记录集合上执行的特殊函数对于每条记录都要在此窗口内执行函数&#xff0c;有的函数随着记录不同&#xff0c;窗口大小都是固定的&#xff0c;这种属于静态窗口…

c语言:枚举类型—enum

枚举类型一.常见形式二.枚举和宏定义三.枚举的意义四.插个小知识一.常见形式 这里举一个例子&#xff0c;我想要枚举颜色 注意一下细节&#xff0c;所有成员间用逗号隔开&#xff0c;最后一个成员后不加标点符号 这里看上去和定义结构体和联合体的样式一样&#xff0c;但其实前…

minio安装部署和minIO-Client的使用

minio安装部署和minIO-Client的使用 一、服务器安装minio 1.进行下载 下载地址&#xff1a; GNU/Linux https://dl.min.io/server/minio/release/linux-amd64/minio2.新建minio安装目录&#xff0c;执行如下命令 mkdir -p /home/minio/data把二进制文件上传到安装目录后&a…

【PAT甲级 - C++题解】1128 N Queens Puzzle

✍个人博客&#xff1a;https://blog.csdn.net/Newin2020?spm1011.2415.3001.5343 &#x1f4da;专栏地址&#xff1a;PAT题解集合 &#x1f4dd;原题地址&#xff1a; 题目详情 - 1128 N Queens Puzzle (pintia.cn) &#x1f511;中文翻译&#xff1a;皇后问题 &#x1f4e3;…

第9章 无线网络和移动网络

目录 9.1 无线局域网 WLAN 9.1.1 无线局域网的组成 1. 无线局域网 WLAN (Wireless Local Area Network) 2. IEEE 802.11 3. 移动自组网络 9.1.2 802.11 局域网的物理层 9.1.3 802.11 局域网的 MAC 层协议 1. CSMA/CA 协议 2. 时间间隔 DIFS 的重要性 3. MAC两个子层…

acwing基础课——Floyd

由数据范围反推算法复杂度以及算法内容 - AcWing 常用代码模板3——搜索与图论 - AcWing 基本思想&#xff1a; floyd算法的原理是基于动态规划的基础上实现的&#xff0c;因为是稠密图我们通过邻接矩阵来存储&#xff0c;我们将各点距离初始化为正无穷(该点到自己的距离为0)&…

软件测试基础理论体系学习8-什么是验收测试?验收测试的内容是什么?过程是什么?有什么测试策略?

8-什么是验收测试&#xff1f;验收测试的内容是什么&#xff1f;过程是什么&#xff1f;有什么测试策略&#xff1f;1 验收测试的主要内容1.1 简介和说明1.2 验收测试的目的1.3 验收测试的任务1.4 验收测试主要内容1.4.1 验收测试标准1.4.2 配置复审1.4.3 α、β测试2 验收测试…

基于intel低功耗平台边缘计算解决方案助力半导体设备升级

半导体芯片是现代电子领域的大脑。事实上&#xff0c;在通信、计算、零售、医疗保健和运输应用领域&#xff0c;半导体芯片为各种先进技术提供了基础。2020年全球半导体销售额增长6.5%&#xff0c;相关制造设备的生产需求也相应增加。 某业内日本半导体设备制造厂商&#xff0…

营销不知道怎么做,不妨试试社交新零售电商结合新型引流模式

大家好&#xff0c;我是林工&#xff0c;如今移动互联网的快速发展为社交媒体分销提供了生根发芽的土壤&#xff0c;人们不仅仅满足于单方面的购物消费&#xff0c;开始利用社交属性&#xff0c;出现了具有强大带货能力和分销能力的人群&#xff0c;也就是当时的代购和微商&…

BEVFormer-accelerate:基于EasyCV加速BEVFormer

作者&#xff1a;贺弘 夕陌 谦言 临在 导言 BEVFormer是一种纯视觉的自动驾驶感知算法&#xff0c;通过融合环视相机图像的空间和时序特征显式的生成具有强表征能力的BEV特征&#xff0c;并应用于下游3D检测、分割等任务&#xff0c;取得了SOTA的结果。我们在EasyCV开源框架&…

【MySQL自学之路】第3天——MySQL数据库服务的基本操作

目录 前言 数据库的初始化 添加环境变量 数据库配置文件的编写【my.ini】 初始化MySQL服务 MySQL服务操作 安装一个MySQL服务 删除一个MySQL服务 开启你的MySQL服务 关闭你的MySQL服务 查看电脑中存在的MySQL服务 其他操作 登录数据库 修改密码 登录跳过密码验证…

Hi3861开发遇到的小问题及解决方法和一些小技巧

开发过程中的学习 一、硬件支持开启&#xff08;替换文件&#xff09; &#x1f604; 在D:\ubuntu2004Harmony镜像中通过vscode 使用编译命令 python build.py wifiiot是存在一个小问题 No option ‘riscv32-unknown-elf-gcc_path’ in section: ndk’ ” 而在E:\鸿蒙资料\鸿蒙…

[附源码]Python计算机毕业设计儿童绘本租赁网站Django(程序+LW)

该项目含有源码、文档、程序、数据库、配套开发软件、软件安装教程 项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等…

基于jsp+mysql+ssm工程车辆租赁系统-计算机毕业设计

项目介绍 随着现代工程车租赁的快速发展&#xff0c;可以说工程车租赁已经逐渐成为现代工程车租赁过程中最为重要的部分之一。但是一直以来我国传统的工程车租赁并没有建立一套完善的行之有效的工程车租赁系统&#xff0c;传统的工程车租赁已经无法适应高速发展&#xff0c;无…

Idea制作docker镜像

一、配置docker远程访问 1.在linux系统中查找docker.service文件&#xff0c;使用命令&#xff1a;find / -name docker.service 2.编辑 /usr/lib/systemd/system/docker.service文件&#xff0c;在ExecStart/usr/bin/dockerd后面添加 -H tcp://0.0.0.0:2375 3.重新加载dock…

持续盈利背后,水滴“新增长”难寻?

配图来自Canva可画 近年来随着互联网红利消失&#xff0c;越来越多的互联网企业回归效益&#xff0c;开始将“降本增效”纳入了议事日程之中&#xff0c;这不仅体现在BAT等一众大厂的财报中&#xff0c;就连水滴等互联网领域的“后起之秀”也在争相效法。 作为过去几年有名的…

最受欢迎的工具-从burp到支付和暴破

文章目录一、封神台七1、认识burp2、功能3、burp字体设置4、尝试抓包/改包&#xff08;支付&#xff09;5、尝试抓包改包&#xff08;暴力破解&#xff09;一、封神台七 1、认识burp 浏览器和相关应用程序的中间拦截、修改、重放数据包的代理工具 2、功能 1、拦截本机8080端…