题目描述
在一个m × n的棋盘的每一格都放有一个礼物,每个礼物都有一定的价值(价值大于 0)。你可以从棋盘的左上角开始拿格子里的礼物,并每次向右或者向下移动一格、直到到达棋盘的右下角。给定一个棋盘及其上面的礼物的价值,请计算你最多能拿到多少价值的礼物?
如输入这样的一个二维数组,
[
[1,3,1],
[1,5,1],
[4,2,1]
]
那么路径 1→3→5→2→1
可以拿到最多价值的礼物,价值为12
示例1
输入:[[1,3,1],[1,5,1],[4,2,1]]
返回值:12
思路 & 解答
这道题其实一看就知道是动态规划,棋盘中的每个小格子,都是和上方,或者左方的格子有关。既然是动态规划,那么我们先定义状态:
dp[i][j]]
: 从(0,0)到(i,j)所能拿到的礼物的最大价值
定义好状态之后,需要找出状态转移方程,也就是如何从前面的结果中,推导到现在的结果,在这道题中,只能向右或者向下,那么假设当前格子的礼物是gift[i][j]
,当前的最大价值dp[i][j]
是Max(dp[i-1][j]+gift[i][j],dp[i][j-1]+gift[i][j])
。
当然如果是第一行和第一列,其实是边界情况,不需要处理,dp[i][j] = gift[i][j]
总结一下:
但是我们可以直接在原数组上操作,不用新建dp
数组。
Java
代码实现如下:
public class Solution47 {
public int maxValue(int[][] gifts) {
int n = gifts.length, m = gifts[0].length;
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
int x = gifts[i][j];
if (i - 1 >= 0) {
gifts[i][j] = Math.max(gifts[i][j], x + gifts[i - 1][j]);
}
if (j - 1 >= 0) {
gifts[i][j] = Math.max(gifts[i][j], x + gifts[i][j - 1]);
}
}
}
return gifts[n - 1][m - 1];
}
}
c++
代码如下:
class Solution {
public:
int maxValue(vector<vector<int> >& gifts) {
int n = gifts.size(), m = gifts[0].size();
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
int x = gifts[i][j];
if (i - 1 >= 0) {
gifts[i][j] = max(gifts[i][j], x + gifts[i - 1][j]);
}
if (j - 1 >= 0) {
gifts[i][j] = max(gifts[i][j], x + gifts[i][j - 1]);
}
}
}
return gifts[n - 1][m - 1];
}
};
- 时间复杂度:
O(nm)
,需要计算完里面的小格子 - 空间复杂度:
O(1)
,优化后可以实现原地操作,不需要额外的空间