面试题47:礼物的最大价值
2019-11-12 本文已影响0人
繁星追逐
在一个mxn的棋盘的每一格斗放油一个礼物,每个礼物都有一定的价值(大于0)
- 从棋盘的左上角开始,每次可以往右边或者下边移动一格,知道到达棋盘的右下角。
- 给定一个棋盘和上面的礼物,计算我们最多可以拿到多少价值的礼物
思路一:
dfs 回溯法
将所有的路径都走一遍,全局变量记录最大值
代码如下:
/**
* 回溯法,递归遍历每个方向的值,保存最大值
* @param gifts
* @param rows
* @param cols
* @return
*/
//全局变量记录最大值,或者使用数组
private int max = 0;
public int getMax(int[] gifts, int rows, int cols) {
if (gifts == null || gifts.length == 0) return -1;
// int[] max = {0};
select(gifts, 0, 0, rows, cols, 0);
return max;
}
private void select(int[] gifts, int row, int col, int rows, int cols, int value) {
if (row >= rows || col >= cols) return;
value += gifts[row*cols + col];
if (row == rows-1 && col == cols-1){
if (value > max) max = value;
}
select(gifts, row+1, col, rows, cols, value);
select(gifts, row, col+1, rows, cols, value);
}
思路二:
- 方法2:动态规划,到达f(i,j)处拥有的礼物价值和有两种情况:
* 1、从左边来,即f(i, j) = f(i, j -1) + gift(i, j)
* 2、从上边来,即f(i, j) = f(i -1, j) + gift(i, j)
*
* 保证到达每一个格子得到的礼物价值之和都是最大的,也就是取max[f(i, j-1), f(i-1, j)] +gift(i, j)
* 可以发现,要知道当前格子能获得最大礼物价值,需要用到当前格子左边一个和上面一个格子的最大礼物价值和
代码如下:
public int getMaxVal(int[] gifts, int rows, int cols) {
if (gifts == null || gifts.length == 0) return 0;
//设立一个记录最大值的二维数组
int[][] maxVal = new int[rows][cols];
for (int row=0;row<rows;row++){
for (int col=0;col<cols;col++){
//申明在外面
int up = 0;
int left = 0;
//必须保证不在起始位置上
if (row > 0) up = maxVal[row-1][col];
if (col > 0) left = maxVal[row][col-1];
maxVal[row][col] = Math.max(up,left) + gifts[row*cols + col];
}
}
return maxVal[rows-1][cols-1];
}