算法-最短路径

news2025/1/18 13:49:54

图的最短路径问题是一个经典的计算机科学和运筹学问题,旨在找到图中两个顶点之间的最短路径。这种问题在多种场景中都有应用,如网络路由、地图导航等。

解决图的最短路径问题有多种算法,其中最著名的包括:
1.迪杰斯特拉算法
(1). 图的要求
适用于权重非负的图。
(2). 实现
该算法的实现通常包括以下步骤:
a. 初始化:将源节点标记为最短路径已经知道.设置其路径距离为0.将其入队列.
b. 循环迭代,直到队列为空
b.1. 出队列,得p
b.2.迭代&更新.
即对从p可达,且最短路径尚未确定节点q
比较,若p的路径距离+Edge<p,q>小于q的路径距离,则更新q的路径距离=p的路径距离+Edge<p,q>.设置p为其路径上一节点.
b.3.从最短路径尚未确定节点中选出路径值最小节点t.将t的最短路径标记为已经知道.t入队列.在无法选出这样的t时,表示剩余节点均不可达.可提前结束迭代.

(3). 实例
在这里插入图片描述
(4). 算法实现

#define MAX 0x7fffffff
class NodeInfo{
public:
    char m_nName;
    int32_t m_nPath = MAX;
    int32_t m_nPreIndex = -1;
    int32_t m_nTag = -1;
};

template<class T>
class Node{
public:
    T m_nEle;
};

template<class EdgeInfo>
class Edge{
public:
    int32_t m_nWeight = MAX;
    bool m_bValid = false;
};

Node<NodeInfo> stNodes[7];
Edge<int> stEdges[7][7];
void Sort(void* lpBegin, void *lpEnd);
int main(){
    stNodes[0].m_nEle.m_nName = 'A';
    stNodes[1].m_nEle.m_nName = 'B';
    stNodes[2].m_nEle.m_nName = 'C';
    stNodes[3].m_nEle.m_nName = 'D';
    stNodes[4].m_nEle.m_nName = 'E';
    stNodes[5].m_nEle.m_nName = 'F';
    stNodes[6].m_nEle.m_nName = 'G';
    
    stEdges[0][1].m_nWeight = 1;
    stEdges[0][1].m_bValid = true;
    stEdges[1][0].m_bValid = true;
    stEdges[1][0].m_nWeight = 1;

    stEdges[0][4].m_bValid = true;
    stEdges[0][4].m_nWeight = 2;

    stEdges[4][1].m_bValid = true;
    stEdges[4][1].m_nWeight = 2;

    stEdges[1][3].m_bValid = true;
    stEdges[1][3].m_nWeight = 1;

    stEdges[1][5].m_bValid = true;
    stEdges[1][5].m_nWeight = 4;
    stEdges[4][3].m_bValid = true;
    stEdges[4][3].m_nWeight = 2;
    stEdges[3][2].m_bValid = true;
    stEdges[3][2].m_nWeight = 1;
    stEdges[3][6].m_bValid = true;
    stEdges[3][6].m_nWeight = 2;
    stEdges[2][6].m_bValid = true;
    stEdges[2][6].m_nWeight = 2;
    
    int nSourceIndex = 0;
    stNodes[nSourceIndex].m_nEle.m_nTag = 1;
    stNodes[nSourceIndex].m_nEle.m_nPath = 0;
    stNodes[nSourceIndex].m_nEle.m_nPreIndex = -1;

    int32_t nArrQueue[7];
    int32_t nFirst = 0;
    int32_t nEnd = 0;
    int32_t nNum = 0;
    nArrQueue[0] = nSourceIndex;
    nFirst = 0;
    nEnd = 1;
    nNum = 1;
    while(nNum > 0){
        // 出队列
        int32_t nIndex;
        if(nNum = 1){
            nIndex = nArrQueue[nFirst];
            nFirst = 0;
            nEnd = 0;
            nNum = 0;
        }
        else{
            nIndex = nArrQueue[nFirst];
            nFirst = (nFirst+1)%7;
            nNum--;
        }

        // 迭代&更新
        for(int32_t i = 0; i < 7; i++){
            if(stEdges[nIndex][i].m_bValid && stNodes[i].m_nEle.m_nTag == -1){
                if(stNodes[i].m_nEle.m_nPath > stNodes[nIndex].m_nEle.m_nPath + stEdges[nIndex][i].m_nWeight){
                    stNodes[i].m_nEle.m_nPath = stNodes[nIndex].m_nEle.m_nPath + stEdges[nIndex][i].m_nWeight;
                    stNodes[i].m_nEle.m_nPreIndex = nIndex;
                }
            }
        }

        // 入队列
        // 选择未访问节点中最短距离最小的一个
        nIndex = -1;
        int32_t nMin = MAX;
        for(int32_t i = 0; i < 7; i++){
            if(stNodes[i].m_nEle.m_nTag == -1 && stNodes[i].m_nEle.m_nPath < nMin){
                nMin = stNodes[i].m_nEle.m_nPath;
                nIndex = i;
            }
        }
        if(nIndex == -1){
            break;// 所有节点均已被访问.或剩余节点全部不可达.
        }
        // 选举的节点就是最短路径已知的
        stNodes[nIndex].m_nEle.m_nTag = 1;
        if(nNum == 0){
            nArrQueue[0] = nIndex;
            nFirst = 0;
            nEnd = 1;
            nNum++;
        }
        else{
            nArrQueue[nEnd] = nIndex;
            nEnd = (nEnd+1)%7;
            nNum++;
        }
    }

    // test
    printf("finish\n");
    while(true){
        int32_t nIndex = -1;
        scanf("%d", &nIndex);
        getchar();
        printf("path_%d\n", stNodes[nIndex].m_nEle.m_nPath);
        printf("%c ", stNodes[nIndex].m_nEle.m_nName);
        while(nIndex != -1){
            printf("%c ", stNodes[stNodes[nIndex].m_nEle.m_nPreIndex].m_nEle.m_nName);
            nIndex = stNodes[nIndex].m_nEle.m_nPreIndex;
        }
        printf("\n");
    }
    return 0;
}

2.贝尔曼-福特算法(Bellman-Ford Algorithm):
(1). 图的要求
可以处理带有负权重的图,但无法处理包含负权重环的图。
针对权重为负的图,可以让所有边权中加上一个基础量转变为权重非负的,再通过迪杰斯特拉求解.所以,正常没必要用这个.
时间复杂度为O(|V|*|E|)

(2). 算法
贝尔曼-福特算法(Bellman-Ford Algorithm)的实现过程可以分为以下三个阶段:

a. 初始化阶段:
创建一个数组Distant,用于记录从源点s到图中各个顶点的最短路径长度估计值。通常,将源点s到自己的距离Distant[s]初始化为0,而将源点s到其他所有顶点的距离初始化为一个较大的值(如无穷大),表示这些顶点与源点之间的最短路径尚未确定。
b. 松弛操作阶段:
这个阶段需要进行|V|-1次迭代,其中V是图中顶点的数量。在每一次迭代中,遍历图中的所有边(u, v),并检查是否可以通过这条边来更新从源点s到顶点v的最短路径长度估计值。
具体来说,对于每一条边(u, v),如果Distant[u] + w(u, v) < Distant[v],则更新Distant[v]Distant[u] + w(u, v)。这里,w(u, v)是边(u, v)的权重,表示从顶点u到顶点v的距离或成本。
通过不断的松弛操作,Distant数组中的值会逐渐逼近从源点到各个顶点的实际最短路径长度。
c. 负权回路检测阶段:
在完成|V|-1次松弛操作后,再进行一次额外的松弛操作。这次操作的目的是为了检测图中是否存在负权回路(即从某个顶点出发,经过一系列边后回到该顶点,且整个回路的总权重为负)。
如果在额外的松弛操作中,仍然有Distant数组的值被更新,那么就说明图中存在负权回路。因为负权回路的存在会导致最短路径问题无解,因为可以通过不断绕行负权回路来减小路径长度。
如果额外的松弛操作没有更新Distant数组的值,那么算法结束,Distant数组中存储的就是从源点到各个顶点的最短路径长度。

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

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

相关文章

抖音小店怎么定类目?分享几个爆单几率大,适合新手的细分类目!

大家好&#xff0c;我是电商糖果 做电商的应该经常听过这么一句话&#xff0c;类目大于一切&#xff01; 好的类目可以让商家减少很多竞争和难题。 糖果做电商有很多年了&#xff0c;我一直认为做店前期最难的定类目&#xff0c;中期是选品&#xff0c;后期是维护店铺。 如…

物联网数据报表分析

随着物联网技术的迅猛发展&#xff0c;越来越多的企业开始将物联网解决方案应用于各个领域&#xff0c;从提高生产效率到优化用户体验&#xff0c;物联网都发挥着至关重要的作用。然而&#xff0c;如何有效地分析和管理物联网产生的海量数据&#xff0c;成为企业面临的挑战之一…

【Java开发过程中的流程图】

流程图由一系列的图形符号和箭头组成&#xff0c;每个符号代表一个特定的操作或决策。下面是一些常见的流程图符号及其含义&#xff1a; 开始/结束符号&#xff08;圆形&#xff09;&#xff1a;表示程序的开始和结束点。 过程/操作符号&#xff08;矩形&#xff09;&#xff…

<Linux> 生产者消费者模型

目录 前言&#xff1a; 一、什么是生产者消费者模型 &#xff08;一&#xff09;概念 &#xff08;二&#xff09;生产者消费者之间的关系 &#xff08;三&#xff09;生产者消费者模型特点 &#xff08;四&#xff09;生产者消费者模型的优点 二、基于阻塞队列实现生产…

《定时执行专家》:Nircmd 的超级搭档,解锁自动化新境界

目录 Nircmd 简介 《定时执行专家》与 Nircmd 的结合 示例&#xff1a; 自动清理电脑垃圾: 定时发送邮件: 定时关闭电脑: 《定时执行专家》的优势: 总结: 以下是一些其他使用示例&#xff1a; 立即下载《定时执行专家》&#xff1a; Nircmd 官方网站&#xff1a; 更…

代码随想录阅读笔记-栈与队列【用队列实现栈】

题目 使用队列实现栈的下列操作&#xff1a; push(x) -- 元素 x 入栈pop() -- 移除栈顶元素top() -- 获取栈顶元素empty() -- 返回栈是否为空 注意: 你只能使用队列的基本操作-- 也就是 push to back, peek/pop from front, size, 和 is empty 这些操作是合法的。你所使用的语言…

普发Pfeiffer镀膜机Classic580-500SP Spider600 全套资料包含操作使用说明

普发Pfeiffer镀膜机Classic580-500SP Spider600 全套资料包含操作使用说明

LeetCode---126双周赛

题目列表 3079. 求出加密整数的和 3080. 执行操作标记数组中的元素 3081. 替换字符串中的问号使分数最小 3082. 求出所有子序列的能量和 一、求出加密整数的和 按照题目要求&#xff0c;直接模拟即可&#xff0c;代码如下 class Solution { public:int sumOfEncryptedInt…

瑞芯微RK3576|触觉智能:开启科技新篇章

更多产品详情可关注深圳触觉智能官网&#xff01; “瑞芯微&#xff0c;创新不止步&#xff01;”——全新芯片RK3576即将震撼登场。指引科技风潮&#xff0c;创造未来无限可能&#xff01;这款芯片在瑞芯微不断创新和突破的道路上&#xff0c;不仅是对过往成就的完美延续&…

ROS2从入门到精通0-3:VSCode 搭建 ROS2 工程环境

目录 0 专栏介绍1 Ubuntu下安装VSCode1.1 基本安装1.2 将VSCode添加到侧边栏 2 VSCode集成相关插件3 VSCode运行ROS2环境步骤3.1 安装编译依赖项3.2 创建工作空间和源码空间3.3 启动VSCode与配置 4 测试工程环境4.1 C版本4.2 Python版本 0 专栏介绍 本专栏旨在通过对ROS2的系统…

【机器学习入门 】逻辑斯蒂回归和分类

系列文章目录 第1章 专家系统 第2章 决策树 第3章 神经元和感知机 识别手写数字——感知机 第4章 线性回归 文章目录 系列文章目录前言一、分类问题的数学形式二、最大似然估计三、交叉熵损失函数四、多类别分类多类别逻辑斯蒂回归归一化指数函数交叉熵误差和均方误差的比较 五…

详解JavaScript中this指向

this 原理 this 是一个指针型变量&#xff0c;它指向当前函数的运行环境。 1.内存的数据结构 var obj { foo: 5 };2.函数 var obj { foo: function () {} };引擎会将函数单独保存在内存中&#xff0c;然后再将函数的地址赋值给foo属性的value属性。 由于函数是一个单独的…

目标检测——PP-YOLO算法解读

PP-YOLO系列&#xff0c;均是基于百度自研PaddlePaddle深度学习框架发布的算法&#xff0c;2020年基于YOLOv3改进发布PP-YOLO&#xff0c;2021年发布PP-YOLOv2和移动端检测算法PP-PicoDet&#xff0c;2022年发布PP-YOLOE和PP-YOLOE-R。由于均是一个系列&#xff0c;所以放一起解…

聚酰亚胺PI材料难于粘接,用什么胶水粘接?那么让我们先一步步的从认识它开始(七): 聚酰亚胺PI薄膜的厚度

聚酰亚胺PI薄膜的厚度 聚酰亚胺&#xff08;PI&#xff09;薄膜的厚度可以根据具体的应用需求而有所不同&#xff0c;通常可以在几个微米&#xff08;μm&#xff09;到几十微米之间。下面是一些常见的聚酰亚胺PI薄膜的厚度范围及其应用&#xff1a; 1.超薄膜&#xff1a; 聚酰…

记录‘No module named ‘notebook.notebookapp‘’导致jupyter打不开的解决方法

最初是因为无法重命名文件的问题&#xff0c;更新了notebook&#xff0c;但是更新之后打不开了 在终端输入 jupyter notebook 报错 File "/Users/maclin/Library/Python/3.8/bin/jupyter-notebook", line 5, in <module> from notebook.notebookapp import ma…

#Linux(SSH软件安装及简单使用)

&#xff08;一&#xff09;发行版&#xff1a;Ubuntu16.04.7 &#xff08;二&#xff09;记录&#xff1a; &#xff08;1&#xff09;终端键入&#xff08;root权限&#xff09;安装 apt-get install openssh-server 安装时遇到报错 E: Could not get lock /var/lib/dpkg/…

网络编程套接字——实现简单的TCP网络程序

目录 1、TCP socket API详解 socket()&#xff1a; bind()&#xff1a; listen(): accept(): connect(): 2、简易的TCP网络程序 TcpServer.hpp TcpClient.cc Main.cc Log.hpp ThreadPool.hpp Task.hpp Init.hpp Daemon.hpp dict.txt Makefile 1、TCP socket A…

基于springboot+vue+Mysql的“智慧食堂”设计与实现

开发语言&#xff1a;Java框架&#xff1a;springbootJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包&#xff1a;…

FCN(全卷积神经网络)

目录 一、什么是FCN 1、FCN简介 2、核心思想 二、代码实现 1、FCN结构介绍 2、ResNet-18提取图像特征 3、11卷积层将通道数变换为类别个数 4、转置卷积还原输入图像的高和宽 5、初始化转置卷积层 6、读取数据集 7、训练 8、预测 三、总结 一、什么是FCN 1、FCN简介…

【XR806开发板试用】使用PWM模块模拟手机呼吸灯提示功能

一般情况下&#xff0c;我们的手机在息屏状态&#xff0c;当收到消息处于未读状态时&#xff0c;会有呼吸灯提醒&#xff0c;这次有幸抽中XR806开发板的试用&#xff0c;经过九牛二虎之力终于将环境搞好了&#xff0c;中间遇到各种问题&#xff0c;在我的另一篇文章中已详细描述…