洛谷 普及组 P1508 Likecloud-吃、吃、吃

题目描述

正处在某一特定时期之中的李大水牛由于消化系统比较发达,最近一直处在饥饿的状态中。某日上课,正当他饿得头昏眼花之时,眼前突然闪现出了一个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,依次下去..直到最顶行。这样并不能得到总能量值最高的一条路,因为其他路可能最底行能量值较低,但后来的能量值特别高。比如样例:

洛谷 普及组 P1508 Likecloud-吃、吃、吃_第1张图片

因此我们没办法确定小新一开始应该走哪一边,但如果知道了走哪条路最终的能量值最高,就能确定往哪个方向走。想知道最终的路线可以这么做:

▶每个点取它上一行的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
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;
} 

 

你可能感兴趣的:(洛谷刷题日记)