最短路径Dijkstra算法的简单实现
时间:2022-07-23
本文章向大家介绍最短路径Dijkstra算法的简单实现,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
最近刷题一连碰到好几道关于最短路径的问题自己一开始用深搜过了之后也就没怎么 管,但是之后的好几道用深搜都超时,之后查了资料才知道这种最短路径的问题一般使用广搜的方法。
而且实现起来有好几种算法,用的最多的就是Dijkstra和Flody这两种算法,这两者的主要区别就是Dijkstra主要用来解决一个初始化的点到所有其他点的所有最短路径,而Flody主要用来解决确定的两点之间所存在的最短路径,今天就先讲解一下Dijkstra算法
假设有n个点,那么Dijkstra算法会进行n-1次循环,每次循环找出原点到其他另外所有相邻的点中最短的一个点,注意这里必须是相邻的点,之后会将这个点放入原点的集合中,因为已经找到该点的最短路径了,之后再一次循环,之后的循环就不单单是查找之前已经找到的点的相邻点中的最短路径了,而是找到之前集合中所有已经找到最短路径的点的最短相邻点,然后判断并选择出其中最短的路径及其点,重复这种操作,最后就能查找到原点到所有其他的点的最短路径了。
其实这里面还用到了几何知识,就比如说三角形的知识,比如说a到b中间如果一开始直接有一条路我们记作a->b,但是之后经过搜索之后发现能通过另外一个点c到达点b,并且路径长还小于之前的路径长即a->b > a->c->b的。
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.io.StreamTokenizer;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.PriorityQueue;
import java.util.Queue;
public class minpath第二版 {
public static int n;//存储所有的点
public static int m;//存储所有的边
public static int map[][];//存储有向图中的所有有向边
public static int visit[];//判断每个点是否已经被访问过
public static int leng[];//最后存储原点到所有其他点的最短路径
public static void main(String[] args) throws IOException {
StreamTokenizer in=new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));
PrintWriter out=new PrintWriter(new OutputStreamWriter(System.out));
in.nextToken();n=(int) in.nval;
in.nextToken();m=(int) in.nval;
List<Integer>list1[]=new ArrayList[n];
for(int i=0;i<n;i++)
list1[i]=new ArrayList<>();//存储每个点的邻接点
map=new int [n][n];//初始化每个数组
visit=new int [n];
leng=new int [n];
for(int i=0;i<n;i++)
visit[i]=0;
for(int i=0;i<n;i++)//初始化数组并赋值为最大的值,原因是之后肯定会有最短路径与之比较并肯定会将之替换
{
leng[i]=Integer.MAX_VALUE;
for(int j=0;j<n;j++)
{
map[i][j]=Integer.MAX_VALUE;
}
}
for(int i=0;i<m;i++)//存入各顶点的相邻结点及其有向路径长
{
in.nextToken();int n1=(int) in.nval;
in.nextToken();int n2=(int) in.nval;
in.nextToken();int n3=(int) in.nval;
map[n1-1][n2-1]=n3;
list1[n1-1].add(n2-1);
}
Queue<node>queue=new PriorityQueue<>(compare);//这个优先队列用来存储每次查找出来的最短的最短的点,并将它抛出,所以循环的终止条件是优先队列为空
queue.add(new node(0,0));
while(!queue.isEmpty())
{
node node1=queue.poll();
if(visit[node1.x]==0)
{
visit[node1.x]=1;
for(int i=0;i<list1[node1.x].size();i++)//查询抛出点的所有邻接点
{
if(map[node1.x][list1[node1.x].get(i)]!=Integer.MAX_VALUE&&visit[list1[node1.x].get(i)]==0)//该邻接点需要不是最大值,就意味着两者之间存在边,并且还没有被访问过
{
if(leng[list1[node1.x].get(i)]>node1.length+map[node1.x][list1[node1.x].get(i)])//判断长度是否与之前的路径短,如果短,则替换
{
leng[list1[node1.x].get(i)]=node1.length+map[node1.x][list1[node1.x].get(i)];
queue.add(new node(list1[node1.x].get(i),node1.length+map[node1.x][list1[node1.x].get(i)] ));//加入该点,重复循环
}
}
}
}
}
for(int i=1;i<n;i++)
out.println(leng[i]);
out.flush();
}
static Comparator<node>compare=new Comparator<node>() {//将各个点按长度从小到大排列
@Override
public int compare(node o1, node o2) {
// TODO Auto-generated method stub
return o1.length-o2.length;
}
};
static class node
{
int x;
int length;
public node(int x,int length) {
// TODO Auto-generated constructor stub
this.x=x;
this.length=length;
}
}
}
- Go语言实现顺序存储的线性表实例
- hashpartitioner-Spark分区计算器
- Could not find artifact com.sun:tools:jar:1.5.0
- Java面试官最爱的volatile关键字
- Golang实现带优先级的channel
- java在指定目录下执行dos命令或者bat文件
- 搭建本地 golang 文档服务器
- 浅析Java中的final关键字
- 大话企业级android读书笔记(一)
- 转--Go语言用堆排序的方法进行一千万个int随机数排序
- go语言,变量引用的用法
- 大话企业级android读书笔记(三)
- Ubuntu 11.10 安装JDK
- Go- Buffer Channel- Select-基础理解
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- PHP合并两个或多个数组的方法
- PHP的mysqli_set_charset()函数讲解
- PHP常见加密函数用法示例【crypt与md5】
- PHP的mysqli_sqlstate()函数讲解
- PHP Laravel中的Trait使用方法
- PHP CURL使用详解
- 解析Python 偏函数用法全方位实现
- PHP读取目录树的实现方法分析
- python suds访问webservice服务实现
- 使用Python-OpenCV消除图像中孤立的小区域操作
- PHP实现数组和对象的相互转换操作示例
- PHP单元测试框架PHPUnit用法详解
- PHP5中使用mysqli的prepare操作数据库的介绍
- php中的explode()函数实例介绍
- ThinkPHP5.0框架验证码功能实现方法【基于第三方扩展包】