leetcode 1937. 扣分后的最大得分「动态规划」「拆项」
创始人
2024-11-20 13:38:23

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 ] dp[i][j] dp[i][j]表示前 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 ) ) dp[i][j] = max(dp[i][j], ar[i][j] + dp[i - 1][k] - abs(k - j)) dp[i][j]=max(dp[i][j],ar[i][j]+dp[i−1][k]−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 dp[i][j] = max(dp[i][j], tr[i][j] + (dp[i-1][k] + k) - j), k <= j dp[i][j]=max(dp[i][j],tr[i][j]+(dp[i−1][k]+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 dp[i][j] = max(dp[i][j], tr[i][j] + (dp[i-1][k] - k) + j), k > j dp[i][j]=max(dp[i][j],tr[i][j]+(dp[i−1][k]−k)+j),k>j

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

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

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

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

class Solution { public:     long long maxPoints(vector>& tr) {         int n = tr.size(), m = tr[0].size();         vectordp(m + 1);         vectorpre(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;     } }; 

相关内容

热门资讯

第五届琶洲算法大赛开启全球报名... 中新社广州5月6日电 (记者 许青青)由广州市政府主办的第五届琶洲算法大赛6日正式启动报名并上线了首...
软件性能测试包含哪些测试内容? 性能测试报告 性能测试是对软件产品在特定条件下的性能进行测试和评估的过程。性能测试的内容可以包括以下...
胜硅来新材料取得金属硅粉用除铁... 国家知识产权局信息显示,河南胜硅来新材料科技有限公司取得一项名为“一种金属硅粉用除铁生产系统”的专利...
工业和信息化部批复开展卫星物联... 工业和信息化部日前正式批复开展卫星物联网业务商用试验,试验期为两年。 据了解,本次获批开展卫星物联网...
月球新矿物“铈嫦娥石”是怎么被... 本文转自【央视新闻客户端】; 近日,我国宣布发现三种月球新矿物:铈嫦娥石、镁嫦娥石、铈镁嫦娥石。其中...