Leetcode 787. K 站中转内最便宜的航班

问题描述

有 n 个城市通过 m 个航班连接。每个航班都从城市 u 开始,以价格 w 抵达 v。
现在给定所有的城市和航班,以及出发城市 src 和目的地 dst,你的任务是找到从 src 到 dst 最多经过 k 站中转的最便宜的价格。 如果没有这样的路线,则输出 -1。

Example

示例 1
输入:
n = 3, edges = [[0,1,100],[1,2,100],[0,2,500]]
src = 0, dst = 2, k = 1
输出: 200
解释:
城市航班图如下



从城市 0 到城市 2 在 1 站中转以内的最便宜价格是 200,如图中红色所示。

示例 2
输入:
n = 3, edges = [[0,1,100],[1,2,100],[0,2,500]]
src = 0, dst = 2, k = 0
输出: 500
解释:
城市航班图如下



从城市 0 到城市 2 在 0 站中转以内的最便宜价格是 500,如图中蓝色所示。

Note

  • n 范围是 [1, 100],城市标签从 0 到 n - 1.
  • 航班数量范围是 [0, n * (n - 1) / 2].
  • 每个航班的格式 (src, dst, price).
  • 每个航班的价格范围是 [1, 10000].
  • k 范围是 [0, n - 1].
  • 航班没有重复,且不存在环路

题目链接:787. K 站中转内最便宜的航班 (难度:中等)

思路

从题目的要求来看,需要我们求出从 src 到 dst 的长度不超过 K + 1 的最短路径。由于增加了路径的长度的约束,我们不能够直接地利用 Dijkstra 算法来求解,但是可以通过对其进行一点点改造来解决这个问题。我们需要做的有以下两步:

  • 将 Dijkstra 中的优先队列换为普通队列,并将搜索的最大层次定为 K + 1,这样我们便解决了路径长度不超过 K + 1 的这个约束条件。
  • 制定好对应边的松弛操作。我们用 dist[i] 来表示从 src 到节点 i 的长度不超过 K + 1 的最小花费。从 src 到 i 的最小花费要高于从 src 借助中转站 j 到 i 的最小花费,则更新 dist[i] = dist[j] + price(j, i)【price(j, i) 代表从 j 到 i 的花费】。这样当 i == dst 时,更新 ans。待搜索结束时,ans 便是最终答案

代码

class Solution {
public:
    //使用广度优先搜索,从起点向终点最多跨越 k + 1 层次
    typedef pair<int, int> pair;
    void BFS(vector<vector<pair>>& graph, int& ans, int src, int dst, int K){
        queue<pair> my_queue;
        vector<int> dist(graph.size(), INT_MAX);
        my_queue.push({src, 0});
        int level = 0;
        while(!my_queue.empty()){
            if(level > K)
                break;
            int len = my_queue.size();
            for(int i = 0;i < len;++i){
                int from = my_queue.front().first;
                int price = my_queue.front().second;
                my_queue.pop();
                for(int j = 0;j < graph[from].size();++j){
                    int to = graph[from][j].first;
                    int fee = graph[from][j].second + price;
                    // 剪枝条件
                    if(dist[to] < fee)
                        continue;
                    dist[to] = fee;
                    if(to == dst){
                        ans = min(ans, fee);
                    }else{
                        my_queue.push({to, fee});
                    }
                    
                }
            }
            ++level;
        }
    }
    int findCheapestPrice(int n, vector<vector<int>>& flights, int src, int dst, int K) {
        vector<int> record;
        int ans = INT_MAX;
        vector<vector<pair>> graph(n, vector<pair>());
        for(int i = 0;i < flights.size();++i){
            graph[flights[i][0]].push_back({flights[i][1], flights[i][2]});
        }
        BFS(graph, ans, src, dst, K);
        return ans == INT_MAX ? -1 : ans;
    }
};

执行结果: 32 ms, 11.3 MB

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,635评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,628评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,971评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,986评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,006评论 6 394
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,784评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,475评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,364评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,860评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,008评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,152评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,829评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,490评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,035评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,156评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,428评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,127评论 2 356