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

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

18小时前

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

原创内容 转载请注明出处

分享给朋友:

相关文章

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

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

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

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

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

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

LeetCode 2222题解析:高效统计"010"与"101"子序列数量的算法优化

LeetCode 2222题解析:高效统计"010"与"101"子序列数量的算法优化

一、题目解读题目要求计算给定字符串中包含"010"或"101"子序列的数量。关键难点在于高效遍历所有子序列,避免重复计算。传统暴力解法时间复杂度O(n^3)会超...

牛客网288555题解题指南:动态规划求解小红的暑假(附代码解析)

牛客网288555题解题指南:动态规划求解小红的暑假(附代码解析)

一、题目解读牛客网288555题要求解决一个组合数学问题:有三位朋友,每天需邀请其中一位参加聚会,但不能连续两天邀请同一位朋友。给定天数n,求满足条件的不同邀请方案总数。题目考察动态规划、状态转移及组...

2018年NOIP货币系统解题报告(洛谷P5020):动态规划与完全背包的巧妙应用

2018年NOIP货币系统解题报告(洛谷P5020):动态规划与完全背包的巧妙应用

一、题目解读2018年NOIP货币系统问题(洛谷P5020)要求给定一组货币面额,判断是否存在一种组合方式,使得所有不超过最大面额的金额都能被表示。例如,若面额集合为{1,3,5},则金额1~8均可被...

洛谷1220题解:动态规划与区间DP优化解法(附代码注释)

洛谷1220题解:动态规划与区间DP优化解法(附代码注释)

一、题目解读洛谷1220题要求计算在n个位置放置灯的情况下,通过关闭连续区间灯并移动至区间端点,使得总耗电量最小。需考虑灯的功率与位置差异,设计高效的算法求解最优策略。二、解题思路1. 动态规划 +...

发表评论

访客

看不清,换一张

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