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

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

7个月前 (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)场景。掌握拓扑排序与动态规划的协同应用,可高效解决此类路径优化问题。

原创内容 转载请注明出处

分享给朋友:

相关文章

IOI 1994 洛谷1216:如何用动态规划高效解决数字三角形问题?附完整代码解析

IOI 1994 洛谷1216:如何用动态规划高效解决数字三角形问题?附完整代码解析

题目重解给定一个由数字组成的三角形结构,从顶部出发,每次可以移动到下方相邻的数字,最终到达底部。我们的目标是找到一条路径,使得路径上经过的数字总和最大。这个问题在实际中有许多应用场景,如最优路径规划、...

从零到一掌握背包问题:洛谷P1164题解精讲,附带优化

从零到一掌握背包问题:洛谷P1164题解精讲,附带优化

题目重解:小A带着m元钱来到餐馆,菜单上有n道菜,每道菜都有确定的价格。现在需要计算出刚好花完m元的点菜方案总数。这个问题看似简单,但当菜品数量增多时,暴力枚举就会变得不可行,需要更高效的算法来解决。...

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

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

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

洛谷P4551题解题报告:图论与Trie树优化异或路径问题的实战解析

洛谷P4551题解题报告:图论与Trie树优化异或路径问题的实战解析

一、题目解读洛谷P4551题要求在一个无向图中,寻找任意两点路径权值异或后的最大值。题目输入为图的边信息(点数n和n-1条边),每条边包含起点、终点及权值。需输出所有路径中权值异或的最大值。问题核心在...

牛客25461题解析:花园喷泉距离优化算法(动态规划+后缀数组解法)

牛客25461题解析:花园喷泉距离优化算法(动态规划+后缀数组解法)

一、题目解读牛客25461题要求计算一个花园中n朵花到两个喷泉的最小距离平方和。用户需输入喷泉坐标(x1,y1)和(x2,y2),以及n朵花的坐标(x,y),通过合理分配每朵花到两个喷泉的距离,使总距...

2022 CSP-J 上升点序(洛谷P8816)解题报告:动态规划求解最长上升序列

2022 CSP-J 上升点序(洛谷P8816)解题报告:动态规划求解最长上升序列

一、题目解读2022年CSP-J题目“上升点序”(洛谷P8816)要求给定一个平面点集,每个点的坐标(x,y)均为整数。题目需要构造一个最长的上升序列,序列中相邻点的坐标满足x和y均严格递增。允许使用...

发表评论

访客

看不清,换一张

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