zoukankan      html  css  js  c++  java
  • [LintCode 1029.] 寻找最便宜的航行旅途(最多经过k个中转站)

    LintCode 1029. 寻找最便宜的航行旅途(最多经过k个中转站)

    题目描述

    有n个城市被一些航班所连接。每个航班 (u,v,w) 从城市u出发,到达城市v,价格为w。

    给定城市数目 n,所有的航班flights。你的任务是找到从起点src到终点站dst的最便宜线路的价格,而旅途中最多只能中转K次。

    如果没有找到合适的线路,返回 -1。

    样例

    样例 1:

    输入:
    n = 3,
    flights = [[0,1,100],[1,2,100],[0,2,500]],
    src = 0, dst = 2, K = 0
    输出: 500
    样例 2:

    输入:
    n = 3,
    flights = [[0,1,100],[1,2,100],[0,2,500]],
    src = 0, dst = 2, K = 1
    输出: 200
    注意事项

    1. 总城市数 n 在 1-100 之间,每个城市被标号为 0 到 n-1。
    2. 航线的总数在 0 到 n * (n - 1) / 2 之间
    3. 每条航线会被以 [出发站,终点站,价格] 的形式展现。
    4. 每条航线的价格都在 1-10000之间。
    5. 中转站的总数限制范围为 0 到 n-1 之间。
    6. 不会有重复或者自环航线出现

    解题思路

    单元最短路径问题,加入一个跳步数限制。
    Dijkstra或者SPFA可解。

    参考代码

    SPFA

    class Solution {
    public:
        /**
         * @param n: a integer
         * @param flights: a 2D array
         * @param src: a integer
         * @param dst: a integer
         * @param K: a integer
         * @return: return a integer
         */
        int findCheapestPrice(int n, vector<vector<int>> &flights, int src, int dst, int K) {
            // write your code here
            if (src == dst) return 0;
            // SPFA
            vector<vector<int>> a;
            vector<int> head(n);
            queue<pair<int,int>> q;
            vector<bool> inq(n);
            vector<int> cost(n);
            constexpr int INF = 0x3f3f3f3f;
            // init
            std::fill(head.begin(), head.end(), -1);
            for (auto&& f : flights) { // add
                a.push_back({f[0], f[1], f[2], head[f[0]]});
                head[f[0]] = a.size() - 1;
            }
            std::fill(inq.begin(), inq.end(), false);
            q.push({src, -1});
            inq[src] = true;
            std::fill(cost.begin(), cost.end(), INF);
            cost[src] = 0;
            while (!q.empty()) {
                auto p = q.front(); q.pop();
                int i = p.first;
                int step = p.second;
                inq[i] = false;
                if (step > K) continue;
                for (int idx = head[i]; idx >= 0; idx = a[idx][3]) {
                    int j = a[idx][1];
                    int tmp = cost[i] + a[idx][2];
                    if (tmp < cost[j] && step+1 <= K) {
                        cost[j] = tmp;
                        if (!inq[j]) {
                            q.push({j, step + 1});
                            inq[j] = true;
                        }
                    }
                }
            }
            if (cost[dst] == INF) return -1;
            return cost[dst];
        }
    };
    
  • 相关阅读:
    如何在SQLite中创建自增字段?
    Windows XP平台下编译boost[1.47及以上]
    智能指针的向下转型
    采用Boost::filesystem操作文件
    CodeSmith访问数据库
    std::string的一些操作
    PDF加入内嵌字体
    悟空和唐僧的对话
    收获和教训的一天配置ds1401
    vxworks的一个changlog
  • 原文地址:https://www.cnblogs.com/zhcpku/p/14285708.html
Copyright © 2011-2022 走看看