【数据结构】最短路径(Dijskra算法)

一.引例

计算机网络传输的问题:

怎样找到一种最经济的方式,从一台计算机向网上所有其他计算机发送一条消息。

抽象为:

给定带权有向图G=(V,E)和源点v,求从v到G中其余各顶点的最短路径。

即:

单源点最短路径问题

给定带权有向图G=(V,E)和源点v$\in$V,求从v到G中其余各顶点的最短路径。

二.最短路径

在非网图中,最短路径是指两顶点之间经历的边数最少的路径。

在网图中,最短路径是指两顶点之间经历的边上权值之和最短的路径。

三.Dijskra算法的基本思想

        设置一个集合S存放已经找到最短路径的顶点,S的初始状态只包含源点v,对vi属于V-S,假设从原点v到vi的有向边为最短路径。以后每求得一条最短路径v,……,vk,就将vk加入到集合S中,并将路径v,……,vk,vi与原来的假设相比较,取路径长度较小者为最短路径。重复上述过程,直到集合V中全部顶点加入到集合S中。

四.Dijskra算法的数据结构

1.图的存储结构:

带权的邻接矩阵存储结构(因为需要频繁的读取边)

2.数组dist[n]:

每个分量dist[i]表示当前所找到的从起始点v到终点vi的最短路径的长度。

初态为:若从v到vi有弧,则dist[i]为弧上权值;否则置dist[i]为$\infty$

3.数组path[n]:

path数组,下标为图每个顶点的编号,数组中的元素为由某个顶点到这个顶点的顶点编号。

初态为:若从v到vi有弧,则path[i]为0;否则path[i]为-1。

最终输出最短路径,依靠path数组。

每次更改dist数组的内容,都会在path数组中更新上一个结点的内容。

4.数组s[n]:

存放源点和已生成的终点,其初态为只有一个源点v。

s数组都初始化为0(源点初始化为1),当该点被放入s集合中,将其置为1。

 五.伪代码

1.初始化数组dist、path和s;

2.while(s中的元素个数<n)

        2.1 在dist[n]中求最小值,其下标为k;

        2.2 输出dist[i]和path[j];

        2.3 修改数组dist和path;

        2.4 将顶点vk添加到数组s中

六.代码实现 

#include <iostream>

using namespace std;

const int MAX_VERTEX=10;

//带权(邻接矩阵)有向图
class MGraph{
private:
    int arc[MAX_VERTEX][MAX_VERTEX];//邻接矩阵
    int vertex[MAX_VERTEX];//存储每个结点的信息
    int vertexNum,arcNum;//实际顶点个数,边的条数
public:
    MGraph(int n,int e);
    void Dijkstra(int start);
    int findMinDist(int dist[],int s[]);
    void display();
    void displayPath(int dist[],int path[],int start,int min);
};

int main(int argc, const char * argv[]) {
    MGraph G(5, 7);
    //G.display();
    G.Dijkstra(0);
    return 0;
}
MGraph::MGraph(int n,int e){
    int p,q,w;
    vertexNum=n;
    arcNum=e;
    for(int i=0;i<n;i++){//初始化邻接矩阵
        for(int j=0;j<n;j++){
            arc[i][j]=-1;//-1表示不可到达
        }
    }
    for(int i=0;i<n;i++){
        vertex[i]=i;
        arc[i][i]=0;
    }
    for(int i=0;i<e;i++){
        cin>>p>>q>>w;
        arc[p][q]=w;
    }
}

void MGraph::Dijkstra(int start){
    int *s=new int[vertexNum];
    int *dist=new int[vertexNum];
    int *path=new int[vertexNum];
    int i,num=0,min;
    for(i=0;i<vertexNum;i++){
        dist[i]=arc[start][i];//初始化距离
        s[i]=0;//初始化集合S
        if(arc[start][i]!=-1){//start到i有路径
            path[i]=start;
        }
        else{
            path[i]=-1;
        }
    }
    s[start]=1;//将源点放入集合S中,1表示在集合中,0表示不在集合中
    num++;//num记录集合S中元素的个数
//    for(int i=0;i<vertexNum;i++){
//        cout<<dist[i]<<" ";
//    }
//    cout<<endl;
    while(num<vertexNum){
        min=findMinDist(dist, s);//dist中查找集合S中不存在的顶点到源点的距离的最小值
        cout<<min<<endl;
        s[min]=1;//将新生成的终点加入到集合S中
        num++;
        for(i=0;i<vertexNum;i++){//更新数组dist和path
            if(s[i]==0&&arc[min][i]!=-1){
                if(dist[i]==-1){
                    dist[i]=dist[min]+arc[min][i];//更新dist数组
                    path[i]=min;//更新path数组
                }
                else if(dist[i]!=-1&&(dist[min]+arc[min][i])<dist[i]){
                    dist[i]=dist[min]+arc[min][i];
                    path[i]=min;//更新path数组
                }
            }
        }
//        for(int i=0;i<vertexNum;i++){
//            cout<<dist[i]<<" ";
//        }
//        cout<<endl;
        displayPath(dist, path, start, min);
    }
    delete [] path;
    delete [] s;
    delete [] dist;
}

int MGraph::findMinDist(int dist[],int s[]){
    int i=0,min=0;
    for(i=0;i<vertexNum;i++){//给min赋初始值
        if(s[i]==0&&dist[i]!=-1){//该顶点不在集合S中
            min=i;
            break;
        }
    }
    for(;i<vertexNum;i++){
        if(s[i]==0){//该顶点不在集合S中
            if(dist[i]!=-1&&dist[min]>dist[i]){
                min=i;
            }
        }
    }
    return min;
}

void MGraph::displayPath(int dist[],int path[],int start,int min){
    int pre=min;
    int p[vertexNum],i=0,j;
    while(pre!=start){
        p[i++]=pre;
        pre=path[pre];
    }
    p[i++]=pre;
    cout<<"(";
    for(j=i-1;j>0;j--){
        cout<<p[j]<<",";
    }
    cout<<p[0]<<")";
    cout<<dist[min]<<endl;
}
void MGraph::display(){
    for(int i=0;i<vertexNum;i++){
        for(int j=0;j<vertexNum;j++){
            cout<<arc[i][j]<<" ";
        }
        cout<<endl;
    }
}

 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

优化问题,详解静态优化

优化问题&#xff0c;尤其静态优化问题&#xff0c;在控制系统设计中随处可见&#xff0c;例如基于燃油经济性和驾驶体验的多目标优化的汽车发动机 MAP 标定&#xff0c;基于性能指标优化的飞行器结构设计参数优化&#xff0c;以实验数据与模型输出匹配为目标的电池 RC 等效电路…

[WP] ISCTF2023 Web 部分题解

圣杯战争!!! 反序列化伪协议读取 where_is_the_flag 环境变量根目录当前目录 绕进你的心里 利用正则最大回溯绕过 easy_website or select 用双写绕过 空格用/**/绕&#xff0c;报错注入 wafr codesystem(ca\t /f*) webinclude 扫描得到index.bak备份文件打开为加密的代码 写…

Linux Spug自动化运维平台本地部署与公网远程访问

文章目录 前言1. Docker安装Spug2 . 本地访问测试3. Linux 安装cpolar4. 配置Spug公网访问地址5. 公网远程访问Spug管理界面6. 固定Spug公网地址 前言 Spug 面向中小型企业设计的轻量级无 Agent 的自动化运维平台&#xff0c;整合了主机管理、主机批量执行、主机在线终端、文件…

SAP_ABAP_编程基础_数据集_创建并填充摘录数据集 / 处理摘录数据集

SAP ABAP 顾问&#xff08;开发工程师&#xff09;能力模型_Terry谈企业数字化的博客-CSDN博客文章浏览阅读494次。目标&#xff1a;基于对SAP abap 顾问能力模型的梳理&#xff0c;给一年左右经验的abaper 快速成长为三年经验提供超级燃料&#xff01;https://blog.csdn.net/j…

【U8+】用友U8修改凭证提示:外部凭证在总账系统中不能修改。

【问题描述】 用友U8中&#xff0c;在总账模块中修改凭证的时候&#xff0c;提示&#xff1a; 外部凭证在总账系统中不能修改。 【原因分析】 在软件中&#xff0c;使用其他模块的情况下&#xff0c; 其他模块生成的凭证都会传到总账模块中&#xff0c;进而这些由其他模块生成…

uniapp设置手机通知权限以及uniapp-push2.0推送

unipush2.0代码 export default function () {// 调用获取用户通知权限setPermissions()// 获取客户端唯一的推送标识&#xff0c;可用于测试uni.getPushClientId({success: (res) > {console.log(res.cid)},fail(err) {console.log(err)}})// 监听推送uni.onPushMessage(r…

windows ssh时出现Bad local forwarding specification的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

TA-Lib学习研究笔记——Cycle Indicators (七)

TA-Lib学习研究笔记——Cycle Indicators &#xff08;七&#xff09; Cycle Indicators 周期指标函数组有HT_DCPERIOD, HT_DCPHASE, HT_PHASOR, HT_SINE, HT_TRENDMODE 。 1.HT_DCPERIOD Hilbert Transform - Dominant Cycle Period 函数名&#xff1a;HT_DCPERIOD 名称&am…

Hertz 整合swagger

文章目录 Swagger安装使用用法项目demoSwagger注释用法通用API信息 swag命令行参数swagger路由配置 Swagger 安装 go get 安装可执行文件需要配合 GOPATH 模式工作。 go get github.com/swaggo/swag/cmd/swag 因为从 Go 1.17 开始&#xff0c;在 go mod 模式下通过 go get 下…

【Web】NISACTF 2022 个人复现

目录 ①easyssrf ②babyupload ③ level-up ④bingdundun~ 明天就新生赛了&#xff0c;练套题保持下手感吧 &#xff08;文章只选取了一部分&#xff09; ①easyssrf 输入/flag 输入file:///fl4g 访问/ha1x1ux1u.php ?filephp://filter/convert.base64-encode/resource/…

uni-app 微信小程序之自定义中间圆形tabbar

文章目录 1. 自定义tabbar效果2. pages新建tabbar页面3. tabbar 页面结构4. tabbar 页面完整代码 1. 自定义tabbar效果 2. pages新建tabbar页面 首先在 pages.json 文件中&#xff0c;新建一个 tabbar 页面 "pages": [ //pages数组中第一项表示应用启动页&#xff…

【小布_ORACLE笔记】Part11-6 RMAN Backups

【小布_ORACLE笔记】Part11-6 RMAN Backups 1.track文件的作用 当做差异性备份时&#xff0c;server process对应的RMAN客户端的server process就不用去每个块每个块的检查&#xff0c;只要到trackfile 里面去读一下&#xff0c;看哪个块改变了就直接把哪个块备份下来&#x…

Java研学-IO流(三)

六 字节流 – 字节输出流系列 OutPutStream体系 1 OutPutStream系列 – 字节输出流 // 表示字节输出流所有类的超类&#xff0c;输出流接受输出字节并将其发送到某个接收器 public abstract class OutputStreamFileOutputStream/BufferedOutputStream 2 FileOutputStream类设…

python 实现链表

链表基础知识 链表是在物理内存中不连续&#xff0c;数据通过链表中的指针来链接到下一个元素。 链表由一系列节点组成&#xff0c;节点在运行时动态生成&#xff0c;节点一般包括两个部分&#xff1a;存储数据的数据域&#xff0c;存储下一个节点的指针域 链表的常用操作&a…

<JavaEE> 什么是线程安全?产生线程不安全的原因和处理方式

目录 一、线程安全的概念 二、线程不安全经典示例 三、线程不安全的原因和处理方式 3.1 线程的随机调度和抢占式执行 3.2 修改共享数据 3.3 关键代码或指令不是“原子”的 3.4 内存可见性和指令重排序 四、Java标准库自带的线程安全类 一、线程安全的概念 线程安全是指…

抑郁症由什么引起?

抑郁症的发生并不是单一原因所导致&#xff0c;而是多种因素相互作用的结果。以下是一些主要的原因&#xff1a; 首先&#xff0c;生物学因素在抑郁症的发病中起到了关键作用。研究显示&#xff0c;抑郁症可能与遗传有关&#xff0c;家族中有患抑郁症的成员会增加个体患病的风…

Android studio Load error:undefined path variables

android stuido 报错 Load error&#xff1a;undefined path variables Gson is undefined 处理方法&#xff1a; 点击进行Sync Project with Gradle Files

了解 ignore_above 参数对 Elasticsearch 中磁盘使用的影响

在 Elasticsearch 中&#xff0c;ignore_above 参数允许你忽略&#xff08;而不是索引&#xff09;长于指定长度的字符串。 这对于限制字段的大小以避免性能问题很有用。 在本文中&#xff0c;我们将探讨 “ignore_above” 参数如何影响 Elasticsearch 中字段的大小&#xff0c…

ESP32 MicroPython WEB蓝牙红外遥控小车⑬

ESP32 MicroPython WEB蓝牙红外遥控小车⑬ 1、蓝牙遥控小车2 、红外遥控小车3 、WEB网页摄像头遥控小车 1、蓝牙遥控小车 实验目的 使用“YQD蓝牙小车”APP控制小车 实验内容 使用小车显示屏显示蓝牙连接情况&#xff0c;开启蓝牙名称为“yqd-car”&#xff0c;并设置连接到小…

Hdoop学习笔记(HDP)-Part.16 安装HBase

目录 Part.01 关于HDP Part.02 核心组件原理 Part.03 资源规划 Part.04 基础环境配置 Part.05 Yum源配置 Part.06 安装OracleJDK Part.07 安装MySQL Part.08 部署Ambari集群 Part.09 安装OpenLDAP Part.10 创建集群 Part.11 安装Kerberos Part.12 安装HDFS Part.13 安装Ranger …