LeetCode 3418.机器人可以获得的最大金币数

LeetCode 3418.机器人可以获得的最大金币数

题目描述

给你一个 m x n 的网格 coins。一个机器人从网格的左上角 (0, 0) 出发,目标是到达网格的右下角 (m - 1, n - 1)。在任意时刻,机器人只能向右或向下移动。

网格中的每个单元格包含一个值 coins[i][j]

  • 如果 coins[i][j] >= 0,机器人可以获得该单元格的金币。
  • 如果 coins[i][j] < 0,机器人会遇到一个强盗,强盗会抢走该单元格数值的 绝对值 的金币。

机器人有一项特殊能力,可以在行程中 最多感化 2 个单元格的强盗,从而防止这些单元格的金币被抢走(即该格子的负数视为 0)。

注意:机器人的总金币数可以是负数。

返回机器人在路径上可以获得的 最大金币数

解题思路

这是一道典型的网格路径 DP + 状态拓展 问题。如果没有任何豁免能力,我们只能简单地计算从左上到右下的最大路径和(允许负数)。但现在多了一个限制:最多可以将 2 个负值单元格变为 0

我们采用 自顶向下的记忆化搜索(递归 + 备忘录),这样写起来比迭代 DP 更直观,也方便处理边界和特殊状态。

状态定义

定义递归函数 dfs(i, j, k) 表示:

从起点 (0, 0) 走到 (i, j),且剩余 k 次豁免机会(即还能将 k 个负数单元格变成 0)时,所能获得的最大路径和。

  • 0 <= i < m0 <= j < n
  • k = 0, 1, 2(因为最多感化 2 个强盗)

最终答案就是 dfs(m-1, n-1, 2)

状态转移

对于当前格子 (i, j),其值为 x = coins[i][j]

  1. 不豁免当前格子

    无论 x 是正是负,直接加上它的值。此时豁免次数不变(k 不变)。

    上一步要么来自上方 (i-1, j),要么来自左方 (i, j-1),因此:

    复制代码
    res1 = max(dfs(i-1, j, k), dfs(i, j-1, k)) + x
  2. 豁免当前格子(仅当 k > 0x < 0

    如果还有豁免机会,并且当前格子是负数,我们可以选择"感化"这个强盗,即该格子对路径和的贡献为 0(而不是 x),但同时消耗一次豁免机会(k-1)。

    上一步来自上方或左方,且上一步使用的豁免次数为 k-1

    复制代码
    res2 = max(dfs(i-1, j, k-1), dfs(i, j-1, k-1))

    注意这里没有 + x,相当于加 0。

最终的 dfs(i, j, k)max(res1, res2)(如果 res2 合法的话)。

边界条件

  • 越界 :当 i < 0j < 0 时,路径不存在,返回 INT_MIN 表示负无穷(不可达)。
  • 起点 (0, 0):没有前驱格子,需要特殊处理:
    • 如果还有豁免机会(k > 0),我们可以选择豁免它(若为负数),即 max(x, 0)
    • 否则(k == 0),只能取原值 x

记忆化

使用一个三维数组 memo[i][j][k] 记录已经计算过的状态,初始化为 INT_MIN

当某个状态被第一次计算后,将其结果存入 memo,后续直接返回,避免重复递归。

代码实现(C++)

cpp 复制代码
class Solution {
public:
    int maximumAmount(vector<vector<int>>& coins) {
        int m = coins.size(), n = coins[0].size();
        // memo[i][j][k] : 走到 (i,j) 剩余 k 次豁免的最大和
        vector memo(m, vector(n, array<int, 3>{INT_MIN, INT_MIN, INT_MIN}));
        
        auto dfs = [&](this auto&& dfs, int i, int j, int k) -> int {
            if (i < 0 || j < 0) return INT_MIN; // 越界不可达
            int x = coins[i][j];
            // 起点
            if (i == 0 && j == 0) {
                return memo[i][j][k] = (k ? max(x, 0) : x);
            }
            int& res = memo[i][j][k];
            if (res != INT_MIN) return res; // 已计算过
            
            // 不豁免当前格子
            res = max(dfs(i - 1, j, k), dfs(i, j - 1, k)) + x;
            // 如果还能豁免且当前格子为负数,尝试豁免
            if (k && x < 0) {
                res = max({res, dfs(i - 1, j, k - 1), dfs(i, j - 1, k - 1)});
            }
            return res;
        };
        
        return dfs(m - 1, n - 1, 2);
    }
};

注意 :代码使用了 C++23 的 this auto&& dfs 语法(显式对象参数),使得 lambda 可以递归调用自身,无需借助 std::function,性能更好。如果编译器不支持 C++23,可以用 std::function 或单独写一个成员函数。

复杂度分析

  • 时间复杂度O(m * n * 3) = O(mn)
    每个状态 (i, j, k) 只会被计算一次,单次转移为 O(1)
  • 空间复杂度O(mn),用于存储记忆化数组。

总结

本题是经典路径 DP 的拓展,关键在于发现"豁免次数"可以作为状态的一维,然后用记忆化搜索清晰地表达出"选择豁免"与"不豁免"两种转移。

这种状态扩展的思路在许多带有"操作次数限制"的网格问题中非常常见(如最多修改 k 次、最多使用 k 次魔法等)

相关推荐
寻寻觅觅☆2 小时前
东华OJ-基础题-31-素数(C++)
开发语言·c++·算法
仟濹2 小时前
【算法打卡day36(2026-04-02 周四)】DFS专项训练3
算法·深度优先
B1acktion2 小时前
2.3.插入排序——像打牌一样整理数组,为什么它对“几乎有序”数据特别友好?
数据结构·算法·排序算法
Mr_Xuhhh2 小时前
C++算法刷题:排序子序列、削减整数、最长上升子序列(二)题解
开发语言·c++·算法
tankeven2 小时前
HJ157 剪纸游戏
c++·算法
迈巴赫车主2 小时前
蓝桥杯 19717 挖矿java
java·开发语言·数据结构·算法·职场和发展·蓝桥杯
airuike1232 小时前
高性能MEMS IMU:机器人自主运动的核心感知中枢
人工智能·算法·机器人
郝学胜-神的一滴2 小时前
PyTorch张量维度操控:transpose与permute深度拆解与实战指南
人工智能·pytorch·python·深度学习·算法·机器学习
未来之窗软件服务2 小时前
SenseVoicecpp ggml-cann.cpp大模型[AI人工智能(七十六)]—东方仙盟
人工智能·算法·sensevoice·仙盟创梦ide·东方仙盟