floyd算法:

解决任意两点间的最短路径的一种算法,可以正确处理有向图或负权的最短路径问题,同时也被用于计算有向图的传递闭包。

为从的只以集合中的节点为中间节点的最短路径的长度。

  1. 若最短路径经过点k,则
  2. 若最短路径不经过点k,则

因此,

在实际算法中,为了节约空间,可以直接在原来空间上进行迭代,这样空间可降至二维。

我的理解为:

folyd算法是每次选定一个点,查看任意两个顶点的距离是否都小于经过这个点之和的距离。

即:假如ABC三个顶点相连,选定C的时候,查AB的距离是否大于 AC + CB 的距离之和,

如果大于说明找到了一个更短的路径,即A->C->B。

下面是我的例子:

floyd算法是一个动态规划的过程,可以上网查下图中中关于它的证明。

代码:

void floyd(MGraph g,EdgeType *des,EdgeType *path)
{
//初始化 des,path
int i,j; for (i=;i<g.numVexs;i++)
{
for (j=;j<g.numVexs;j++)
{
*(des + i*MAXVEX +j) = g.Mat[i][j];
*(path + i*MAXVEX +j) = j;
}
} int k;
for (k=;k<g.numVexs;k++)
{
for (i=;i<g.numVexs;i++)
{
for (j=;j<g.numVexs;j++)
{
//printf("[%d][%d]---%d [%d][%d]+[%d][%d]---%d\n", i,j,*(des +i*MAXVEX +j),i,k,k,j,g.Mat[i][k] + g.Mat[k][j]);
if ( *(des +i*MAXVEX +j) > *(des +i*MAXVEX +k) + *(des + k*MAXVEX + j))
{ *( des + i*MAXVEX +j ) = *(des +i*MAXVEX +k) + *(des + k*MAXVEX + j);
*( path + i*MAXVEX + j ) = *(path+i*MAXVEX +k);
}
}
}
if (k == )
{
break;
}
}
}

第一个for是初始化,第二个for快里面嵌入2个for循环,三个for循环完成最短路径的查找,算法很神奇。

调用后,我给出的结果是一个矩阵形式:

在Dest中,表示v0(A)到其他点的最短距离,和之前的Dijkstra算法对顶点A的结果一致。其他行类推。

在Edge中,由于点初始化的时候,每个点写入的是自己,这样在查看最短路径的时候,这样看

比如查找v0->v8的最小路径,查看第一行(v0)的第八行(v8):显示是1,表示到v8点,需要经过v1,

再查看第二行(v1)到v8的值:2,,表示v1到v8需要经过v2。。。依次类推。

v0到v8的路径为:

v0->1->v2->v4->v7->v8

可以写一个代码来显示最短路径:

void prt_short_path(int vs,int ve,EdgeType *p)
{
int x = vs;
printf(" %c --> ",g_init_vexs[x]);
while(x != ve)
{
x = *(p + MAXVEX*x + ve);
printf(" %c --> ",g_init_vexs[x]);
}
}

完整代码:

// grp-dijkstra.cpp : 定义控制台应用程序的入口点。
// #include "stdafx.h"
#include <stdlib.h>
#include <string.h> #define MAXVEX 100
#define IFY 65535 typedef char VertexType;
typedef int EdgeType; bool g_visited[MAXVEX]; VertexType g_init_vexs[MAXVEX] = {'A','B','C','D','E','F','G','H','I'}; EdgeType g_init_edges[MAXVEX][MAXVEX] = {
{,,,IFY,IFY,IFY,IFY,IFY,IFY}, //'A'
{,,,,,IFY,IFY,IFY,IFY}, //'B'
{,,,IFY,,,IFY,IFY,IFY},//'C'
{IFY,,IFY,,,IFY,,IFY,IFY},//'D'
{IFY,,,,,,,,IFY}, //'E'
{IFY,IFY,,IFY,,,IFY,,IFY}, //'F'
{IFY,IFY,IFY,,,IFY,,,}, //'G'
{IFY,IFY,IFY,IFY,,,,,}, //'H'
{IFY,IFY,IFY,IFY,IFY,IFY,,,}, //'I'
}; //静态图-邻接矩阵
typedef struct {
VertexType vexs[MAXVEX];
EdgeType Mat[MAXVEX][MAXVEX];
int numVexs,numEdges;
}MGraph; //====================================================================
//打印矩阵
void prt_maxtix(EdgeType *p,int vexs)
{
int i,j;
for (i=;i<vexs;i++)
{
printf("\t");
for (j=;j<vexs;j++)
{
if( (*(p + MAXVEX*i + j)) == IFY)
{
printf(" $ ");
}
else
{
printf(" %2d ", *(p + MAXVEX*i + j));
}
}
printf("\n");
}
} //check the number of vextex
int getVexNum(VertexType *vexs)
{
VertexType *pos = vexs;
int cnt=;
while(*pos <= 'Z' && *pos >= 'A')
{
cnt++;
pos++;
}
return cnt;
} bool checkMat(EdgeType *p,VertexType numvex)
{
int i,j;
for (i=;i<numvex;i++)
{
for(j=i+;j<numvex;j++)
{
//printf("[%d][%d] = %d\t",i,j,*(p + MAXVEX*i + j));
//printf("[%d][%d] = %d\n",j,i,*(p + MAXVEX*j + i));
if (*(p + MAXVEX*i + j) != *(p + MAXVEX*j +i) )
{
printf("ERROR:Mat[%d][%d] or Mat[%d][%d] not equal!\n",i,j,j,i);
return false;
}
}
}
return true;
} void init_Grp(MGraph *g,VertexType *v,EdgeType *p)
{
int i,j;
// init vex num
(*g).numVexs = getVexNum(v); //init vexter
for (i=;i<(*g).numVexs;i++)
{
(*g).vexs[i]=*v;
v++;
} //init Mat
for (i=;i<(*g).numVexs;i++)
{
for (j=;j<(*g).numVexs;j++)
{
(*g).Mat[i][j] = *(p + MAXVEX*i + j);
}
}
if(checkMat(&((*g).Mat[][]),(*g).numVexs) == false)
{
printf("init error!\n");
getchar();
exit();
}
} void init_path(EdgeType *p,int num)
{
int i,j;
for (i=;i<num;i++)
{
for (j=;j<num;j++)
{
*(p + i*MAXVEX + j) = j;
}
}
} void floyd(MGraph g,EdgeType *des,EdgeType *path)
{
//初始化 des,path
int i,j; for (i=;i<g.numVexs;i++)
{
for (j=;j<g.numVexs;j++)
{
*(des + i*MAXVEX +j) = g.Mat[i][j];
*(path + i*MAXVEX +j) = j;
}
} int k;
for (k=;k<g.numVexs;k++)
{
for (i=;i<g.numVexs;i++)
{
for (j=;j<g.numVexs;j++)
{
//printf("[%d][%d]---%d [%d][%d]+[%d][%d]---%d\n", i,j,*(des +i*MAXVEX +j),i,k,k,j,g.Mat[i][k] + g.Mat[k][j]);
if ( *(des +i*MAXVEX +j) > *(des +i*MAXVEX +k) + *(des + k*MAXVEX + j))
{ *( des + i*MAXVEX +j ) = *(des +i*MAXVEX +k) + *(des + k*MAXVEX + j);
*( path + i*MAXVEX + j ) = *(path+i*MAXVEX +k);
}
}
} }
}
void prt_short_path(int vs,int ve,EdgeType *p)
{
int x = vs;
printf(" %c --> ",g_init_vexs[x]);
while(x != ve)
{
x = *(p + MAXVEX*x + ve);
printf(" %c --> ",g_init_vexs[x]);
}
}
int _tmain(int argc, _TCHAR* argv[])
{
MGraph grp; EdgeType edgeDest[MAXVEX][MAXVEX];
EdgeType edgePath[MAXVEX][MAXVEX]; int i; init_Grp(&grp,g_init_vexs,&g_init_edges[][]);
prt_maxtix(&grp.Mat[][],grp.numVexs);
printf("Floyd start!\n");
floyd(grp,&edgeDest[][],&edgePath[][]); printf("Dest:\n");
prt_maxtix(&edgeDest[][],grp.numVexs); printf("Edge:\n");
prt_maxtix(&edgePath[][],grp.numVexs); prt_short_path(,,&edgePath[][]);
printf("finish\n");
getchar();
return ;
}

算法学习记录-图——最小路径之Floyd算法的更多相关文章

  1. 算法学习记录-图——最小生成树之prim算法

    一个连通图的生成树是一个极小的连通子图,它包含图中全部的顶点(n个顶点),但只有n-1条边. 最小生成树:构造连通网的最小代价(最小权值)生成树. prim算法在严蔚敏树上有解释,但是都是数学语言,很 ...

  2. 算法学习->求解三角形最小路径及其值

    00 问题 00-1 描述 对给定高度为n的一个整数三角形,找出从顶部到底部的最小路径和.每个整数只能向下移动到与之相邻的整数. 找到一个一样的力扣题:120. 三角形最小路径和 - 力扣(LeetC ...

  3. 算法学习->求解三角形最小路径

    00 问题 00-1 描述 对给定高度为n的一个整数三角形,找出从顶部到底部的最小路径和.每个整数只能向下移动到与之相邻的整数. 找到一个一样的力扣题:120. 三角形最小路径和 - 力扣(LeetC ...

  4. 算法学习记录-图——最短路径之Dijkstra算法

    在网图中,最短路径的概论: 两顶点之间经过的边上权值之和最少的路径,并且我们称路径上的第一个顶点是源点,最后一个顶点是终点. 维基百科上面的解释: 这个算法是通过为每个顶点 v 保留目前为止所找到的从 ...

  5. 算法学习记录-图——最小生成树之Kruskal算法

    之前的Prim算法是基于顶点查找的算法,而Kruskal则是从边入手. 通俗的讲:就是希望通过 边的权值大小 来寻找最小生成树.(所有的边称为边集合,最小生成树形成的过程中的顶点集合称为W) 选取边集 ...

  6. 算法学习记录-图(DFS BFS)

    图: 目录: 1.概念 2.邻接矩阵(结构,深度/广度优先遍历) 3.邻接表(结构,深度/广度优先遍历) 图的基本概念: 数据元素:顶点 1.有穷非空(必须有顶点) 2.顶点之间为边(可空) 无向图: ...

  7. 算法学习记录-图——应用之关键路径(Critical Path)

    之前我们介绍过,在一个工程中我们关心两个问题: (1)工程是否顺利进行 (2)整个工程最短时间. 之前我们优先关心的是顶点(AOV),同样我们也可以优先关心边(同理有AOE).(Activity On ...

  8. 算法学习记录-图——应用之拓扑排序(Topological Sort)

    这一篇写有向无环图及其它的应用: 清楚概念: 有向无环图(DAG):一个无环的有向图.通俗的讲就是从一个点沿着有向边出发,无论怎么遍历都不会回到出发点上. 有向无环图是描述一项工程或者系统的进行过程的 ...

  9. 图的最短路径---弗洛伊德(Floyd)算法浅析

    算法介绍 和Dijkstra算法一样,Floyd算法也是为了解决寻找给定的加权图中顶点间最短路径的算法.不同的是,Floyd可以用来解决"多源最短路径"的问题. 算法思路 算法需要 ...

随机推荐

  1. 创建对象js.

    JavaScript中的基本书记类型. Number(数值类型) String(字符串类型) boolean(布尔类型) null(空类型) undefined(未定义类型) object 常见的内置 ...

  2. 阻塞 io 非阻塞 io 学习笔记

    阻塞 io 非阻塞 io 学习笔记

  3. 数据库 join

    数据库 join 最近有个项目用到了数据库,写 sql 时有联表查询,不知道怎么写,怎过滤,查了一些资料,补充了一些知识点. 以下是链接: 维基百科 JOIN关联表中ON,WHERE后面跟条件的区别 ...

  4. 一行JS搞定快速关机

    一.在本地新建一个文件js文件 JS代码: (new ActiveXObject("Shell.Application")).ShutdownWindows(); 二.设置快捷键 ...

  5. webstorm使用总结

    1.webstorm显示ES6语法错误,和nodejs语法错误未提示的问题,只需要在 此处解决ES6语法错误问题: 此处解决不支持node语法的问题: 然后就显示正常啦.

  6. KVC/KVO 本质

    KVO 的实现原理 KVO是关于runtime机制实现的 当某个类的对象属性第一次被观察时,系统就会在运行期动态地创建该类的一个派生类,在这个派生类中重写基类中任何被观察属性的setter方法.派生类 ...

  7. Spark中Java函数的使用方法笔记

    1: map 函数map是对RDD中的每个元素都执行一个指定的函数来产生一个新的RDD. 任何原RDD中的元素在新RDD中都有且只有一个元素与之对应. 2: mapPartitions函数</p ...

  8. jmeter中通过jdbc方式连接mysql数据库的配置参考

    jmeter中通过jdbc方式连接mysql数据库的配置参考: Database URL=jdbc:mysql://ip:port/dbname?useUnicode=true&allowMu ...

  9. 通过90行代码学会HTML5 WebSQL的4种基本操作

    Web SQL数据库API是一个独立的规范,在浏览器层面提供了本地对结构化数据的存储,已经被很多现代浏览器支持了. 我们通过一个简单的例子来了解下如何使用Web SQL API在浏览器端创建数据库表并 ...

  10. java 核心技术卷一笔记 6 .1.接口 lambda 表达式 内部类

    6.1.2 接口不是类,不能实例化一个接口:但是可以声明接口的变量:Comparable x;    接口变量必须引用实现了接口的类对象:x = new Employee(); 检查一个对象是否属于某 ...