当前位置:首页 > 洛谷 > 洛谷P1137题解:拓扑排序与动态规划求解城市游览问题

洛谷P1137题解:拓扑排序与动态规划求解城市游览问题

1个月前 (08-04)

洛谷P1137题解:拓扑排序与动态规划求解城市游览问题 洛谷题解 拓扑排序算法 动态规划 有向图 邻接表 图论 第1张

一、题目解读

洛谷P1137题目要求解决一个基于有向图的旅行计划问题:给定N个城市和M条单向道路,每个城市可游览一次,求每个城市最多能游览多少个连续城市(包括自身)。问题本质是寻找拓扑排序下的最长路径,需利用图论算法动态规划思想。

二、解题思路

核心思路是将问题转化为拓扑排序与动态规划的结合:

1. 构建有向:用邻接表存储道路信息,记录每个节点的入度。

2. 拓扑排序:优先处理入度为0的节点,确保遍历顺序正确。

3. 动态规划更新:在拓扑排序过程中,每个节点的最大游览数由其前驱节点的最大值+1决定。

三、解题步骤

1. 输入与初始化:读取N、M,构建邻接表graph与入度数组inDegree,初始化dp数组为1(每个城市至少游览自己)。

2. 入度处理:将入度为0的节点加入队列q,作为拓扑排序起点。

3. 拓扑排序循环:

○ 弹出队首节点u,遍历其邻居v。

○ 更新dp[v] = max(dp[v], dp[u] + 1),记录最长路径。

○ 若v的入度减为0,则加入队列继续处理。

4. 输出结果:遍历dp数组输出每个城市的最长游览数。

四、代码与注释

#include <iostream>
#include <queue>
#include <vector>
using namespace std;

int main() {
    int N, M;
    cin >> N >> M;
    // 构建图的邻接表表示
    vector<vector<int>> graph(N + 1); // 1-based索引
    vector<int> inDegree(N + 1, 0);   // 记录每个节点的入度

    // 读取道路信息并构建图
    for (int i = 0; i < M; ++i) {
        int x, y;
        cin >> x >> y;
        graph[x].push_back(y); // x -> y 的有向边
        inDegree[y]++;         // y的入度增加
    }

    // 初始化拓扑排序队列
    queue<int> q;
    vector<int> dp(N + 1, 1); // 每个城市至少能游览自己

    // 将入度为0的节点加入队列
    for (int i = 1; i <= N; ++i) {
        if (inDegree[i] == 0) {
            q.push(i);
        }
    }

    // 拓扑排序
    while (!q.empty()) {
        int u = q.front();
        q.pop();

        // 遍历u的所有邻居
        for (int v : graph[u]) {
            // 更新v的最大游览城市数
            dp[v] = max(dp[v], dp[u] + 1);

            // 减少v的入度,如果入度为0则加入队列
            if (--inDegree[v] == 0) {
                q.push(v);
            }
        }
    }

    // 输出结果
    for (int i = 1; i <= N; ++i) {
        cout << dp[i] << endl;
    }

    return 0;
}

五、总结

该解法巧妙利用拓扑排序保证遍历顺序,结合动态规划实时更新最优解,时间复杂度为O(N+M),适用于有向无环图(DAG)场景。掌握拓扑排序与动态规划的协同应用,可高效解决此类路径优化问题。

原创内容 转载请注明出处

分享给朋友:

相关文章

2024年GESP五级武器强化(洛谷B4071)解题代码C++版

2024年GESP五级武器强化(洛谷B4071)解题代码C++版

一、题目解读    2024年GESP(青少年软件编程能力等级考试)五级中的“武器强化”(洛谷平台题目编号B4071)是一道典型的算法优化问题。题目要求通过合理...

CSP-J方格取数题解|动态规划解法|洛谷P7074代码解析

CSP-J方格取数题解|动态规划解法|洛谷P7074代码解析

一、题目解读题目要求在一个n×m的网格中,从左上角到右下角选择一条路径,路径上的数字可重复取用,求取数之和的最大值。路径限制为仅能向右或向下移动。需注意路径的灵活性与重复取数的可能性,传统单向动态规划...

牛客12576题解题全解析:动态规划+质因数分解实现跳跃问题最优解

牛客12576题解题全解析:动态规划+质因数分解实现跳跃问题最优解

一、题目解读牛客12576题是一道经典的算法题,要求给定起点N和终点M,求解从N到M的最少跳跃次数。题目考察的核心在于路径优化与动态规划思想,需结合数论中的质因数分解技巧,通过合理设计算法降低时间复杂...

NOIP 2008火柴棒等式题解(C++代码实现)  动态规划与枚举算法详解

NOIP 2008火柴棒等式题解(C++代码实现) 动态规划与枚举算法详解

一、题目解读火柴棒等式问题(NOIP 2008,洛谷P1149)要求使用给定数量的火柴棒,构造形如 A + B = C 的等式,其中A、B、C均为整数,且火柴棒总数恰好等于输入值。需统计符合条件的等式...

LeetCode 120题三角形最小路径和最优解法:动态规划详解与代码实现

LeetCode 120题三角形最小路径和最优解法:动态规划详解与代码实现

一、题目解读LeetCode 120题“三角形最小路径和”要求给定一个由数字组成的三角形,从顶部开始向下移动,每次可向左或向右移动一格,计算从顶至底的最小路径和。三角形以二维向量形式给出,每层元素数量...

CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解

CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解

一、题目解读CSP-J 2019年的“公交换乘”题目(洛谷P5661)要求模拟地铁与公交交替出行的费用计算。题目核心在于地铁消费会产生优惠券,而公交可在45分钟内使用优惠券抵扣车费。需要处理n条出行记...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。