当前位置:首页 > 洛谷 > 洛谷P3800题解:动态规划与单调队列优化的高效解法

洛谷P3800题解:动态规划与单调队列优化的高效解法

8个月前 (08-14)

洛谷P3800题解:动态规划与单调队列优化的高效解法 洛谷题解 动态规划 单调队列 滑动窗口算法 C++ 第1张

一、题目解读

洛谷P3800题要求在一个N×M的网格中,每个点有能量值,从第一行出发到第N行,每次可向左/右移动不超过T个位置,求路径上能量值总和的最大值。题目核心在于如何高效处理路径限制与能量累积,考验动态规划与优化技巧的应用。

二、解题思路

参考代码采用动态规划+单调队列优化。定义dp[i][j]为从起点到(i,j)位置的最大能量和,利用单调队列维护每一行中“可到达列”的递减能量值序列,确保滑动窗口内始终取到最优前缀。通过双向处理(正向+反向维护队列),兼顾左右移动限制,降低时间复杂度至O(NM),突破暴力枚举的瓶颈。

三、解题步骤

1. 数据读取与初始化:存储K个特殊点的坐标与能量值,初始化第一行dp值(即直接取网格能量)。

2. 动态规划核心逻辑:

○ 正向处理:维护列j的单调递减队列,确保队首为当前可覆盖范围内(j-T,j)的最大dp值,更新dp[i][j] = 队首值 + 当前网格能量。

○ 反向处理:同理维护队列,覆盖范围变为(j,j+T),取双向计算结果中的最大值。

3. 结果输出:遍历最后一行dp值,输出最大值。

四、代码与注释

#include <iostream>
#include <vector>
#include <deque>
#include <algorithm>
using namespace std;

typedef long long ll;
const int MAX_N = 5005;
const int MAX_M = 5005;

struct Point {
    int x, y, v;
};

int main() {
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    
    int N, M, K, T;
    cin >> N >> M >> K >> T;
    
    vector<vector<ll>> grid(N+1, vector<ll>(M+1, 0));
    vector<vector<ll>> dp(N+1, vector<ll>(M+1, 0));
    
    // 读取P点数据
    for (int i = 0; i < K; ++i) {
        int x, y, v;
        cin >> x >> y >> v;
        grid[x][y] = v;
    }
    
    // 初始化第一行
    for (int j = 1; j <= M; ++j) {
        dp[1][j] = grid[1][j];
    }
    
    // 动态规划处理每一行
    for (int i = 2; i <= N; ++i) {
        deque<int> dq;
        
        // 处理每一列,使用单调队列优化
        for (int j = 1; j <= M; ++j) {
            // 维护单调队列,保持队列中dp[i-1][k]递减
            while (!dq.empty() && dp[i-1][dq.back()] <= dp[i-1][j]) {
                dq.pop_back();
            }
            dq.push_back(j);
            
            // 移除超出T范围的列
            while (!dq.empty() && dq.front() < j - T) {
                dq.pop_front();
            }
            
            // 计算当前列的最大值
            dp[i][j] = dp[i-1][dq.front()] + grid[i][j];
        }
        
        // 反向处理,考虑从右边移动过来的情况
        dq.clear();
        for (int j = M; j >= 1; --j) {
            while (!dq.empty() && dp[i-1][dq.back()] <= dp[i-1][j]) {
                dq.pop_back();
            }
            dq.push_back(j);
            
            while (!dq.empty() && dq.front() > j + T) {
                dq.pop_front();
            }
            
            // 取两个方向的最大值
            dp[i][j] = max(dp[i][j], dp[i-1][dq.front()] + grid[i][j]);
        }
    }
    
    // 输出最后一行中的最大值
    ll max_power = *max_element(dp[N].begin(), dp[N].end());
    cout << max_power << endl;
    
    return 0;
}

五、总结

本题通过动态规划构建状态转移方程,结合单调队列优化,巧妙将二维路径问题转化为一维滑动窗口求解。关键在于双向队列维护的边界判断,既保证了路径限制的合法性,又通过单调性避免了冗余计算。该解法为处理带区间限制的优化问题提供了经典范式。

原创内容 转载请注明出处

分享给朋友:

相关文章

力扣746:三步通关最小花费爬楼梯

力扣746:三步通关最小花费爬楼梯

题目解析:站在楼梯的某个台阶时,需要支付当前台阶对应的体力值cost[i],之后可以选择向上爬1或2个台阶。最终目标是到达‌楼层顶部‌(即数组末尾之后的位置),且初始位置可选择下标0或1的台阶作为起点...

征服力扣704题:三步掌握经典二分查找算法

征服力扣704题:三步掌握经典二分查找算法

题目重解我们面对的是算法领域最经典的二分查找问题:在一个已排序的整数数组中,快速定位目标值的位置。就像在一本按字母顺序排列的字典中查找单词,我们不需要逐页翻阅,而是通过不断折半的方式快速缩小搜索范围,...

GESP2023年六级真题解析:动态规划解决小杨买饮料问题(洛谷3873)

GESP2023年六级真题解析:动态规划解决小杨买饮料问题(洛谷3873)

一、题目解读小杨买饮料是GESP 2023年六级认证考试中的一道经典动态规划题目,考察学生对背包问题的理解和应用能力。题目描述小杨需要购买n种饮料,每种饮料有特定的体积w和价格v,他要在不超过容量l的...

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

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

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

CSP-J 2019纪念品题解(洛谷P5662):动态规划+完全背包问题的实战应用

CSP-J 2019纪念品题解(洛谷P5662):动态规划+完全背包问题的实战应用

一、题目解读2019年CSP-J的“纪念品”问题(对应洛谷P5662)要求玩家在T天内通过买卖纪念品最大化金币收益。每天可交易N种商品,需计算最优策略下的最终金币数。题目强调动态规划思维与资源分配优化...

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

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

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

发表评论

访客

看不清,换一张

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