Skip to content

Latest commit

 

History

History
96 lines (68 loc) · 2.8 KB

47.礼物的最大价值.md

File metadata and controls

96 lines (68 loc) · 2.8 KB

47.礼物的最大价值

题目描述

在一个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),优化后可以实现原地操作,不需要额外的空间