题目信息
示例:
1 2 3 4 5 6 7 8
| 输入: [ [1,3,1], [1,5,1], [4,2,1] ] 输出: 12 解释: 路径 1→3→5→2→1 可以拿到最多价值的礼物
|
注意
1 2
| 1. 0 < grid.length <= 200 2. 0 < grid[0].length <= 200
|
解题思路
本题难点
根据题目说明,某单元格可能从上边单元格或左边单元格到达。
具体思路
动态规划解决此问题,转移方程f(i,j)=max[f(i,j−1),f(i−1,j)]+grid(i,j)
设动态规划矩阵 dp(i,j) 代表从棋盘的左上角开始,到达单元格 (i,j) 时能拿到礼物的最大累计价值。
- 当 i = 0 且 j = 0时,起始元素。
- 当 i = 0 且 j != 0时,为矩阵第一行元素,只可从左边到达;
- 当 i != 0 且 j = 0时,为矩阵第一列元素,只可从上边到达;
- 当 i != 0 且 j != 0时,可从左边或上边到达;
dp(i,j)=⎩⎪⎪⎨⎪⎪⎧grid(i,j)grid(i,j)+dp(i,j−1)grid(i,j)+dp(i−1,j)grid(i,j)+max[dp(i−1,j),dp(i,j−1)],i=0,j=0,i=0,j=0,i=0,j=0,i=0,j=0
注意:由于 dp[i] [j] 只与 dp[i−1] [j] , dp[i] [j−1] , grid[ i ] [ j ]有关系,因此可以将原矩阵 grid 用作 dp 矩阵,即直接在 grid 上修改即可。
代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| class Solution { public int maxValue(int[][] grid) { int m = grid.length, n = grid[0].length; for(int i = 0; i < m; i++) { for(int j = 0; j < n; j++) { if(i == 0 && j == 0) continue; if(i == 0) grid[i][j] += grid[i][j - 1] ; else if(j == 0) grid[i][j] += grid[i - 1][j]; else grid[i][j] += Math.max(grid[i][j - 1], grid[i - 1][j]); } } return grid[m - 1][n - 1]; } }
|
复杂度分析:
- 时间复杂度 O(MN) : M,N分别为矩阵行高、列宽;动态规划需遍历整个 grid 矩阵。
- 空间复杂度 O(1) : 原地修改使用常数大小的额外空间。
其他优秀解答
解题思路
多开辟一个二维数组的空间,节省边界值的判断。
代码
1 2 3 4 5 6 7 8 9 10 11 12 13
| class Solution { public int maxValue(int[][] grid) { int row = grid.length; int col = grid[0].length; int[][] dp = new int[row+1][col+1]; for(int i = 1;i <= row;i++){ for(int j = 1; j <= col; j++){ dp[i][j] = Math.max(dp[i-1][j],dp[i][j-1]) + grid[i-1][j-1]; } } return dp[row][col]; } }
|