带你找到五一最省的旅游路线【dijkstra算法推导详解】

前言

五一快到了,小张准备去旅游了!
查了查到各地的机票

  
因为今年被扣工资扣得很惨,小张手头不是很宽裕,必须精打细算。他想弄清去各个城市的最低开销。
【嗯,不用考虑回来的开销。小张准备找警察叔叔说自己被拐卖,免费被送回来。】
如果他想从珠海飞到拉萨,最少要花多少机票钱呢?下面就说到我们今天要说的这个算法。

迪杰斯特拉(Dijkstra)算法

Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个节点到其他所有节点的最短路径。主要特点是以起始点为中心向外层层扩展,直到扩展到终点为止。Dijkstra算法的时间复杂度为O(N^2)。

扩展

狄克斯特拉Dijkstra1930年5月11日生于荷兰鹿特丹的一个知识分子家庭,在兄弟姊妹4人中排行第三。他的父亲是一名化学家和发明家,曾担任荷兰化学会主席。他母亲则是一位数学家。他成功地设计并实现了在有障碍物的两个地点之间找出一条最短路径的高效算法,这个算法被命名为“狄克斯特拉算法”,解决了机器人学中的一个十分关键的问题,即运动路径规划问题,至今仍被广泛应用。

算法推导

做个表来记录珠海到各个城市的最少机票开销

我们开始找从珠海直达的城市

珠海直达的城市有上海、北京、广州、重庆,那么珠海到其他城市的机票价格如下(无法直达的我们标记无穷大):

可以看出,这4个城市中广州价格最低,那我们就从广州转机吧

从机票最便宜的广州转机

广州能直达的城市有北京、拉萨,那么珠海从广州转机到达其他城市的机票价格如下:(无法知道就能从广州转机)

对比发现从珠海到广州 200 ,广州到北京600,算下来才800块钱(可能时间花销上损失,管他呢,小张穷的只剩下时间了)
从广州中转,到拉萨1700,那么肯定比到不了强。
这么算下来我们有最便宜的价格表了。

除了广州,那再从我们再找转机最便宜的城市--上海

上海直达的城市重庆、南京,那么珠海从上海转机到达其他城市的机票价格如下:

对比原来的价格,发现上海中转到重庆、南京比较便宜

除了广州、上海,那再从我们再找转机最便宜的城市--北京

北京直达上海(上海已经被标记了,肯定已经是最便宜的价格,其实已经没有比较的意义)、杭州和拉萨,价格如下:

到拉萨的价格 即 到北京最低的价格800 + 北京 -> 拉萨 1400 的价格之和(2200)高于1700,到杭州 800 + 500 = 1300,那么最低价格表如下

除了广州、上海、北京,那再从我们再找转机最便宜的城市--南京

南京只能直达杭州,

南京到杭州的价格为1100,划算

除了广州、上海、北京、南京,那再从我们再找转机最便宜的城市--重庆

重庆直达的只有南京,且到南京需要1000 + 400 = 1400元,和原来的到南京的800比,肯定不合算

除了广州、上海、北京、南京、重庆,那再从我们再找转机最便宜的城市--杭州

杭州也只能到上海,且比上海价格高

最终找到拉萨


那么拉萨最便宜的机票就是1700元。

代码实现

变量准备

1)用0,1,2,. . . ,7分别表示珠海,上海,北京,广州,重庆,南京,杭州,拉萨。
2)用一个二维数组 prices [8][8] 来表示航班价格:prices[i][j] = i到j的直飞价格(如无航班记作∞)
3)用一个数组minPrice来记录珠海到各个城市的最少机票开销:
4)用一个数组flag标记城市是否已经转机过


    //    表示无穷大 即不可达
    public static int NO_AIRPLANE = Integer.MAX_VALUE;
//    初始直飞价格表
    public int[][]  prices ;
    //    最优转机价格表
    public int[]   minPrice ;
    public boolean[] flag ;
    private int citySize;

数据准备

 public static int[][] getPrices(){
        int ZH = 0,SH = 1, BJ = 2, GZ = 3,CQ = 4,NJ = 5, HZ = 6,LS  = 7;
        int[][] prices =  new int[8][8];
        //from Zhuhai
        prices[ZH][CQ] = 1100;
        prices[ZH][SH] = 600;
        prices[ZH][BJ] = 900;
        prices[ZH][GZ] = 200;
        //others
        prices[CQ][NJ] = 400;
        prices[SH][CQ] = 400;
        prices[SH][BJ] = 500;
        prices[SH][NJ] = 200;
        prices[BJ][SH] = 400;
        prices[BJ][HZ] = 500 ;
        prices[BJ][LS] = 1400;
        prices[GZ][BJ] = 600 ;
        prices[GZ][LS] = 1500 ;
        prices[NJ][HZ] = 300 ;
        prices[HZ][SH] = 200 ;
        for(int i = 0 ; i < 8 ; i++){
            for(int j = 0 ; j < 8 ; j++){
                if(prices[i][j] == 0){
                    prices[i][j] =  NO_AIRPLANE;
                }
            }
        }
        return prices;
    }

初始化杭州直飞的价格

//            初始化始发站价格表
        for(int i = 1; i < citySize;i++){
            minPrice[i-1] = prices[0][i];
        }

算法实现

private void dijkstra(){
        int min = Integer.MAX_VALUE;
        int minIdx = Integer.MAX_VALUE;
//        找到最小的价格
        for(int idx = 0 ; idx < minPrice.length ; idx ++ ) {
            if(!flag[idx] &&  minPrice[idx] < min ){
                min = minPrice[idx];
                minIdx =  idx ;
            }
        }
        if(minIdx == Integer.MAX_VALUE){
//            已经没有最小的了
            return ;
        }
        //标记从该城市转机
        flag[minIdx] = true;
        minIdx += 1;
        System.out.println("最小城市序号"+minIdx +" 价格"+ minPrice[minIdx -1]);
 //        获取当前城市的价格表
        int cityPrice =  minPrice[minIdx -1];
        int[] minCityPrices = prices[minIdx];
        for(int idx = 1 ; idx < citySize ; idx ++ ){
            int price = minCityPrices[idx];
//            如果从杭州到达该城市的价格 加上 idx城市转机的价格 低于  从杭州到达idx城市的价格 则更新
            if(!flag[idx -1 ] && price != NO_AIRPLANE  && (cityPrice+ price) < minPrice[idx - 1]){
//            可达的城市到达的
                minPrice[idx - 1] = cityPrice+ price;
                System.out.println(idx+"更新最优表:" + Arrays.toString(minPrice));
            }
        }
        dijkstra();
    }

运行结果


跟上述推到过程一致

源码下载

待上传

原文地址:https://www.cnblogs.com/Halburt/p/10767389.html

时间: 2024-10-01 05:30:16

带你找到五一最省的旅游路线【dijkstra算法推导详解】的相关文章

Android中的普通对话框、单选对话框、多选对话框、带Icon的对话框、以及自定义Adapter和自定义View对话框详解

对话框就是一个AlertDialog,但是一个简单的AlertDialog,我们却可以将它玩出许多花样来,下面我们就来一起总结一下AlertDialog的用法.看看各位童鞋在平时的工作中否都用到了AlertDialog的这些特性. OK,废话不多说,进入我们今天的正题. 普通对话框 普通对话框就是我们最最常用的对话框,实现起来并不复杂,实现出来的效果当然也是最简单的,如下: AlertDialog dialog = new AlertDialog.Builder(this).setTitle("

java利用myeclipse自带三大框架搭建三大框架(Hibernate+Struts2+Spring)过程详解

搭建过程因人而异,我的搭建过程大致是这样的: 1.创建一个javaweb项目: 2.导入Spring框架,上图: 2.1: 2.2: 2.3: 3.导入struts2框架,上图: 3.1: 3.2: next: 3.3: 4.导入Hibernate框架,说明:由于hibernate属于持久层,和数据库密切相关,所以需要我们提前出创建好数据库对应视图,然后再开始下面的操做.上图: 4.1: 4.2: 4.3: 4.4: 4.5: 4.6: 4.7:利用数据库相关表和hibernate的orm生成实

OSChina 周四乱弹 —— 十一精品旅游路线

国庆第一天,程序员们都是怎么庆祝的呢?有没有兴奋地高歌一曲? @blindcat:大盘向下流哇,手中的股票栽跟头哇,说割咱就割呀,你割我割全是肉.路见到底抄一手哇,抄后发现想剁手哇,疯疯癫癫想跳楼哇.嘿呀~依儿呀~真嘿真嘿他娘呀,看见抄底一声吼哇,朋友千万别出手哇,精神病院住满楼哇! 看来还有人没有从前段时间的股灾里恢复过来 假期还是抽个时间好好学习吧 @铂金小猎:今天遇到个乞丐,老叫我买书,烦死了 你买了乞丐的书,就会变成下面这样 @blindcat: 本名:工人阶级:假名:中国的领导阶级:经

详解SpringMVC请求的时候是如何找到正确的Controller[附带源码分析]

目录 前言 源码分析 重要接口介绍 SpringMVC初始化的时候做了什么 HandlerExecutionChain的获取 实例 资源文件映射 总结 参考资料 前言 SpringMVC是目前主流的Web MVC框架之一. 如果有同学对它不熟悉,那么请参考它的入门blog:http://www.cnblogs.com/fangjian0423/p/springMVC-introduction.html 我们使用浏览器通过地址 http://ip:port/contextPath/path进行访问

有向网络(带权的有向图)的最短路径Dijkstra算法

什么是最短路径? 单源最短路径(所谓单源最短路径就是只指定一个顶点,最短路径是指其他顶点和这个顶点之间的路径的权值的最小值) 什么是最短路径问题? 给定一带权图,图中每条边的权值是非负的,代表着两顶点之间的距离.指定图中的一顶点为源点,找出源点到其它顶点的最短路径和其长度的问题,即是单源最短路径问题. 什么是Dijkstra算法? 求解单源最短路径问题的常用方法是Dijkstra(迪杰斯特拉)算法.该算法使用的是贪心策略:每次都找出剩余顶点中与源点距离最近的一个顶点. 算法思想 带权图G=<V,

《21个项目玩转深度学习:基于TensorFlow的实践详解》高清带标签PDF版本学习下载

1 写在前面 <21个项目玩转深度学习——基于TensorFlow的实践详解>以实践为导向,深入介绍了深度学习技术和TensorFlow框架编程内容. 通过本书,读者可以训练自己的图像识别模型.进行目标检测和人脸识别.完成一个风格迁移应用,还可以使用神经网络生成图像和文本,进行时间序列预测.搭建机器翻译引擎,训练机器玩游戏.全书共包含21个项目,分为深度卷积网络.RNN网络.深度强化学习三部分.读者可以在自己动手实践的过程中找到学习的乐趣,了解算法和编程框架的细节,让学习深度学习算法和Tens

Java集合详解6:这次,从头到尾带你解读Java中的红黑树

<Java集合详解系列>是我在完成夯实Java基础篇的系列博客后准备开始写的新系列. 这些文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.com/h2pl/Java-Tutorial 喜欢的话麻烦点下Star.fork哈 文章首发于我的个人博客: www.how2playlife.com 什么是红黑树 首先,什么是红黑树呢? 红黑树是一种"平衡的"二叉查找树,它是一种经典高效的算法,能够保证

详解C++引用——带你走进引用的世界

 一.介绍引用 首先说引用是什么,大家可以记住,引用就是一个别名,比如小王有个外号叫小狗,他的妈妈喊小狗回家吃饭,那就是在喊小王回家吃饭. 接下来我们用两行代码来声明一个引用(就拿小王和小狗来说吧): int xiaoW; int &xiaoG=xiaoW; 上面就是一个引用,说明几点要注意的地方: 1.&不是取地址符,而是引用运算符: 2.xiaoG是xiaoW的别名,所以这两个变量的值和地址都是一样的: 3.引用只能初始化,而不能先声明再赋值,因为引用就相当于一个常量: 4.在声明

AME_Thoy_Oracle自带AME审批链详解AME Standard Handler

Oracle 自带了3大类,13个子类的审批链Action Type, 对应了13个标准的AME Standard Handler 1. 按主管层次审批 absolute job level / chains of authority based on absolute job levelfinal approver only / chains of authority containing only the final job-level approvermanager then final