本文记录一下dijkstra算法的实现,图用邻接矩阵表示,假设图为无向图。而且连通,有向图,不连通图的做法相似。

算法简述:

  1. 首先确定“单源”的源。假设是第0个顶点。

  2. 维护三个数组dist[], color[], path[]。设其下标分别为0…i…n-1:

      dist[] 表示源点到顶点i的最短距离,在初始化时,假设源点到顶点i有路径,则初始化为路径的权重。否则初始化为INT_MAX。

      color[] 数组事实上表示两个集合,即color[i]值为1的集合表示已经确定最短路径的点的集合,color[i]值为0表示没有确定最短路径的点的集合。初始化为将源点的color设置为1,其余点设置为0。

      path[]数组存储到顶点i的路径,假设path[i]=3,path[3]=2,paht[2]=0,则这条最短路径是0->2->3->i,与数组给出的顺序是逆序。

  3. 依次从dist[]数组中选一个最小的dist值,假设顶点的坐标为index,这个dist值即为终于确定的最短距离的点,更新这个点的color值为1。以下一个操作是dijkstra算法的重点。也仅仅有这么一个重点操作,即:在没有确定最短距离的集合中(即color值为0的点的集合),假设源点到index的距离,加上index到这些点的距离小于原来的dist值,则更新dist值,同一时候更新path值。

  4. 反复第3个操作,直到color值为0的集合为空。

以下给出c语言实现代码,方法都出了凝视,这里不再说明, 假设不足之处请提出(使用的默认的图例如以下):

#include <stdlib.h>
#include <stdio.h>
#include <limits.h> int n;
int source = 0; //求从第0个节点到其它节点的最短路径
int* dist;
int* path;
int* color; //颜色为1说明已经找到最短路径。为-1说明没找到最短路径 //获得默认的图,即上图所看到的。使用邻接矩阵表示
int** get_graph(){
int** matrix;
int i,j;
int start,end,weight; printf("input vertex num:\n");
scanf("%d",&n); matrix = (int**)malloc(sizeof(int*)*n); for(i=0;i<n;i++){
matrix[i] = (int*)malloc(sizeof(int)*n);
for(j=0;j<n;j++){
if(i!=j)
matrix[i][j] = INT_MAX;
else
matrix[i][j] = 0;
}
} printf("input start end weight, stop by -1\n"); for(;;){
scanf("%d",&start);
if(start==-1){
break;
}
scanf("%d %d",&end,&weight);
matrix[start][end] = weight;
matrix[end][start] = weight;
}
return matrix;
} //使用迪杰斯特拉算法求单源最短路径
void single_source_shortest_path(int** matrix,int source){ int i,j,index,min; dist = (int*)malloc(sizeof(int)*n);
color = (int*)malloc(sizeof(int)*n);
path = (int*)malloc(sizeof(int)*n); //初始化最短路径:
//直接相连的初始化为权重,不直接相连的初始化为INT_MAX
for(i=0;i<n;i++){
dist[i] = matrix[source][i];
color[i] = 0;
if(i!=source && dist[i]!=INT_MAX){
path[i] = source;
}else{
path[i] = -1;
}
} color[source] = 1;
path[source] = 0; //找一个从源点到其它节点最短的路径
for(j=0;j<n;j++){
min = INT_MAX;
index = -1;
for(i=0;i<n;i++){
if(!color[i] && dist[i]<min){
index = i;
min = dist[i];
}
} if(index==-1){ //全部定点的终于距离都确定
break;
} color[index] = 1; //标记为已经确定最短距离的定点 //接下来更新到每一个未确定最短距离的定点的距离
//假设源点到刚刚加入的节点的最短距离+刚刚加入的节点的距离到未确定最短距离的定点的距离 < 源最短距离,则更新
for(i=0;i<n;i++){
if(!color[i] && matrix[index][i]!=INT_MAX && dist[index]+matrix[index][i]<dist[i]){
dist[i] = dist[index]+matrix[index][i];
path[i] = index;
}
}
}
} int main(){ int** matrix = get_graph();
int i,t; single_source_shortest_path(matrix,source); printf("\n");
for(i=0;i<n;i++){
printf("%d: %d,and the path is(inverse order): %d ",i,dist[i],i);
t = path[i];
while(1){
printf(" %d ",t);
if(t==0){
break;
}
t = path[t];
}
printf("\n");
} printf("\n");
return EXIT_SUCCESS;
}

执行结果例如以下:

单源最短路径 dijkstra算法实现的更多相关文章

  1. 单源最短路径Dijkstra算法,多源最短路径Floyd算法

    1.单源最短路径 (1)无权图的单源最短路径 /*无权单源最短路径*/ void UnWeighted(LGraph Graph, Vertex S) { std::queue<Vertex&g ...

  2. 单源最短路径——dijkstra算法

    dijkstra算法与prim算法的区别   1.先说说prim算法的思想: 众所周知,prim算法是一个最小生成树算法,它运用的是贪心原理(在这里不再证明),设置两个点集合,一个集合为要求的生成树的 ...

  3. 单源最短路径——Dijkstra算法学习

    每次都以为自己理解了Dijkstra这个算法,但是过没多久又忘记了,这应该是第4.5次重温这个算法了. 这次是看的胡鹏的<地理信息系统>,看完之后突然意识到用数学公式表示算法流程是如此的好 ...

  4. 单源最短路径-Dijkstra算法

    1.算法标签 贪心 2.算法描述 具体的算法描述网上有好多,我觉得莫过于直接wiki,只说明一些我之前比较迷惑的. 对于Dijkstra算法,最重要的是维护以下几个数据结构: 顶点集合S : 表示已经 ...

  5. [数据结构与算法-15]单源最短路径(Dijkstra+SPFA)

    单源最短路径 问题描述 分别求出从起点到其他所有点的最短路径,这次主要介绍两种算法,Dijkstra和SPFA.若无负权优先Dijkstra算法,存在负权选择SPFA算法. Dijkstra算法 非负 ...

  6. matlab练习程序(单源最短路径Dijkstra)

    图的相关算法也算是自己的一个软肋了,当年没选修图论也是一大遗憾. 图像处理中,也有使用图论算法作为基础的相关算法,比如图割,这个算法就需要求最大流.最小割.所以熟悉一下图论算法对于图像处理还是很有帮助 ...

  7. 单源最短路径---Bellman-Ford算法

    传送门: Dijkstra Bellman-Ford SPFA Floyd 1.Dijkstra算法的局限性 像上图,如果用dijkstra算法的话就会出错,因为如果从1开始,第一步dist[2] = ...

  8. 洛谷P3371单源最短路径Dijkstra版(链式前向星处理)

    首先讲解一下链式前向星是什么.简单的来说就是用一个数组(用结构体来表示多个量)来存一张图,每一条边的出结点的编号都指向这条边同一出结点的另一个编号(怎么这么的绕) 如下面的程序就是存链式前向星.(不用 ...

  9. 单源最短路径Dijkstra和优先级算法

    百度百科:迪杰斯特拉算法. 代码实现如下: import java.util.Comparator; import java.util.PriorityQueue; import java.util. ...

随机推荐

  1. Node.js 回调函数

    Node.js 回调函数 Node.js 异步编程的直接体现就是回调. 异步编程依托于回调来实现,但不能说使用了回调后程序就异步化了. 回调函数在完成任务后就会被调用,Node 使用了大量的回调函数, ...

  2. javascript循环---性能优化

    循环是编程中是最为常见的结构,优化循环是性能优化中很重要的一个部分. 减值迭代:大多数循环使用一个从0开始.增加到某个特定值的迭代器.在很多情况下,从最大值开始,在循环中不断减值的迭代器更加高效. 简 ...

  3. Spring IOC容器分析(1) -- BeanFactory

    搭建好源码阅读环境后,就可以慢慢走进Spring殿堂了.IOC是Inversion of Control的缩写,控制反转的意思.很多人可能都知道IOC是spring的核心,将对象的创建初始化等权限交由 ...

  4. 原生js实现简单移动端轮播图

    最近项目不是很忙,自己就用原生js写了一个简单的移动端轮播图的小demo,可实现自动轮播和手势滑动轮播,然后就把它记录到个人博客里.还有很多不足的地方,希望多多指出,以便改进. 1.代码部分 分为四个 ...

  5. java与OC比较

    转自:http://blog.sina.com.cn/s/blog_93742d0d010165qi.html 1.Cocoa是什么?Cocoa是使用OC语言编写的工具包,里面有大量的类库.结构体,说 ...

  6. vue 2 仿IOS 滚轮选择器 从入门到精通 (一)

    大家好,由于最近从事的是微信公众号和APP内嵌 H5开发,避免不了开发一些和native相同的操作功能,就如接下来说的 仿IOS滚轮选择器. 先来个截图: 接下来具体介绍如何实现的.能力有限避免不了错 ...

  7. C#中抽象类与接口的区别

    1.面向接口编程和面向对象编程是什么关系 首先,面向接口编程和面向对象编程并不是平级的,它并不是比面向对象编程更先进的一种独立的编程思想,而是附属于面向对象思想体系,属于其一部分.或者说,它是面向对象 ...

  8. 引用reference作用域scope闭包closure上下文context用法

    引用(reference).作用域(scope).闭包(closure)以及上下文(context)是JavaScript重中之重的基础,也是学习好JavaScript的基础.在这里我以浅显的理解给大 ...

  9. STM32学习方法

    1.网络学习资源 WWW.openedv.com      开源电子网 WWW.stmcu.org         ST中国官方技术网站,ST官方文档发布网站 微信公众平台             正 ...

  10. iPhone X 网页导航概念

     以下内容由Mockplus团队翻译整理,仅供学习交流,Mockplus是更快更简单的原型设计工具.   在移动应用程序设计中,选择汉堡菜单按钮还是标签栏作为导航一直是个古老的争论话题.目前看来,由于 ...