题目描述
正处在某一特定时期之中的李大水牛由于消化系统比较发达,最近一直处在饥饿的状态中。某日上课,正当他饿得头昏眼花之时,眼前突然闪现出了一个n*m(n,m≤200)的矩型的巨型大餐桌,而自己正处在这个大餐桌的一侧的中点下边。餐桌被划分为了n*m个小方格,每一个方格中都有一个圆形的巨型大餐盘,上面盛满了令李大水牛朝思暮想的食物。李大水牛已将餐桌上所有的食物按其所能提供的能量打了分(有些是负的,因为吃了要拉肚子),他决定从自己所处的位置吃到餐桌的另一侧,但他吃东西有一个习惯——只吃自己前方或左前方或右前方的盘中的食物。
由于李大水牛已饿得不想动脑了,而他又想获得最大的能量,因此,他将这个问题交给了你。
每组数据的出发点都是最后一行的中间位置的下方!
输入格式
[输入数据:]
第一行为m n.(n为奇数),李大水牛一开始在最后一行的中间的下方
接下来为m*n的数字距阵.
共有m行,每行n个数字.数字间用空格隔开.代表该格子上的盘中的食物所能提供的能量.
数字全是整数.
输出格式
[输出数据:]
一个数,为你所找出的最大能量值.
输入输出样例
输入
6 7
16 4 3 12 6 0 3
4 -5 6 7 0 0 2
6 0 -1 -2 3 6 8
5 3 4 0 0 -2 7
-1 7 4 0 7 -5 6
0 -1 3 4 12 4 2
输出
41
思路
动态规划。把李大水牛当作小新..小新一开始站在固定位置(如图),每次移动只有3个方向:左前方、正前方、右前方。
▶现在小新要从最底行吃到最顶行,那么一开始就可以走到能量值为 3 或 4 或 12 的食物那里
▶一种错误的思路是:从最底行挑能量值最大的路走,即到达12,然后下一步继续找能量值最高的食物,即倒数第二行的7,依次下去..直到最顶行。这样并不能得到总能量值最高的一条路,因为其他路可能最底行能量值较低,但后来的能量值特别高。比如样例:
因此我们没办法确定小新一开始应该走哪一边,但如果知道了走哪条路最终的能量值最高,就能确定往哪个方向走。想知道最终的路线可以这么做:
▶每个点取它上一行的3个方向里值最大的那个数和当前点的值相加,就能确定当前点到最顶行(包括当前点)的路线的最大能量值。比如取第 2 行第 1 列的4,经过处理dp[2][1] += max(dp[1][0],dp[1][1],dp[1][2])就变成4+16=20,即从下面走到 4 这一步之后应该往 16 的方向去。以此类推,每一行的每个点都从上一行的3个方向里找最大值相加,这样遍历到最底行时就能知道走哪个点之后得到的能量值最大
▶从最底行的那3个点取最大值,就是要求的答案
#include<iostream>
using namespace std;
#define mx 202
int n, m, dp[mx][mx];
int max(int x, int y, int z){
int t;
x > y ? t=x: t=y;
t > z ? t=t: t=z;
return t;
}
int main()
{
cin >> n >> m;
for(int i = 1; i <= n; i++)
for(int j = 1; j <= m; j++){
cin >> dp[i][j];
dp[i][j] += max(dp[i-1][j-1], dp[i-1][j], dp[i-1][j+1]);
}
cout << max(dp[n][m/2], dp[n][m/2+1], dp[n][m/2+2]) << endl;
return 0;
}