leetcode 1937. 扣分后的最大得分「动态规划」「拆项」

1937. 扣分后的最大得分

题目描述:

给你一个n*m的整数矩阵ar,一开始你的得分为0,你想最大化从矩阵中得到的分数

你的得分方式为:每一行 中选取一个格子,选中坐标为 (r, c) 的格子会给你的总得分 增加 points[r][c]

然而,相邻行之间被选中的格子如果隔得太远,你会失去一些得分。对于相邻行 rr + 1 (其中 0 <= r < m - 1),选中坐标为 (r, c1)(r + 1, c2) 的格子,你的总得分 减少 abs(c1 - c2)

请你返回你能得到的 最大 得分

思路:

很经典的拆项思路

d p i j dpij dpij表示前 i − 1 i-1 i−1行均选择了一个格子,第 i i i行选了第 j j j个格子,获得的最大价值

很显然状态要从上一行转移过来

  • d p i j = m a x ( d p i j , a r i j + d p i − 1 k − a b s ( k − j ) ) dpij = max(dpij, arij + dpi - 1k - abs(k - j)) dpij=max(dpij,arij+dpi−1k−abs(k−j))

如果这样暴力转移,则时间复杂度是 O ( n ∗ m ∗ m ) O(n*m*m) O(n∗m∗m),一定会超时的

我们考虑拆项,根据 k k k和 j j j的大小,分情况讨论

  • d p i j = m a x ( d p i j , t r i j + ( d p i − 1 k + k ) − j ) , k < = j dpij = max(dpij, trij + (dpi-1k + k) - j), k <= j dpij=max(dpij,trij+(dpi−1k+k)−j),k<=j
  • d p i j = m a x ( d p i j , t r i j + ( d p i − 1 k − k ) + j ) , k > j dpij = max(dpij, trij + (dpi-1k - k) + j), k > j dpij=max(dpij,trij+(dpi−1k−k)+j),k>j

对于 d p i j dpij dpij我们如果可以用 O ( 1 ) O(1) O(1)的时间得到 m a x { d p i − 1 k + k } , k < = j max\{dpi-1k + k\},k<=j max{dpi−1k+k},k<=j 和 m a x { d p i − 1 k − k } max\{dpi-1k-k\} max{dpi−1k−k}的值就行了

所以我们记录一个前缀最大值数组和一个后缀最大值数组,分别记录即可

同时,我们可以发现转移的时候只和上一维度的值有关,可以用滚动数组优化空间

所以最后的时间复杂度是 O ( n ∗ m ) O(n*m) O(n∗m)

cpp 复制代码
class Solution {
public:
    long long maxPoints(vector<vector<int>>& tr) {
        int n = tr.size(), m = tr[0].size();
        vector<long long>dp(m + 1);
        vector<long long>pre(m + 2), suf(m + 2);
        for(int j = 1; j <= m + 1; ++j){
            pre[j] = j;
            suf[j] = -j;
        }
        for(int i = 1; i <= n; ++i){   
            for(int j = 1; j <= m; ++j){
                dp[j] = tr[i - 1][j - 1] + max(pre[j] - j, suf[j + 1] + j);
            }
            for(int j = 1; j <= m; ++j){
                pre[j] = max(pre[j - 1], dp[j] + j);
            }
            for(int j = m; j >= 1; --j){
                suf[j] = max(suf[j + 1], dp[j] - j);
            }
        }
        long long ans = 0;
        for(int i = 1; i <= m; ++i)ans = max(ans, dp[i]);
        return ans;
    }
};
相关推荐
8Qi85 小时前
LeetCode 213:打家劫舍 II(House Robber II)—— 题解 ✅
算法·leetcode·职场和发展·动态规划
Lsk_Smion6 小时前
力扣实训 _ [75].颜色分类 _ 杨辉三角
数据结构·算法·leetcode
8Qi88 小时前
LeetCode 1049:最后一块石头的重量 II —— 题解 ✅
算法·leetcode·职场和发展·动态规划·01背包
运筹vivo@11 小时前
LeetCode 2574. 左右元素和的差值
算法·leetcode·职场和发展·每日一题
一只齐刘海的猫12 小时前
【Leetcode】移动零
算法·leetcode·职场和发展
workflower12 小时前
医院核心竞争力的四大重构
人工智能·安全·设计模式·重构·动态规划·scrum
人道领域12 小时前
【LeetCode刷题日记】131.分割回文串,动态规划优化
java·开发语言·leetcode
汉克老师14 小时前
GESP6级C++考试语法知识(五十三、动态规划----背包问题(六、分组背包)
c++·动态规划·背包问题·gesp6级·gesp六级·分组背
Lsk_Smion15 小时前
力扣实训 _ [994].腐烂的橘子/图论
算法·leetcode·图论
8Qi815 小时前
LeetCode 337:打家劫舍 III(House Robber III)—— 题解 ✅
算法·leetcode·二叉树·动态规划