275. 传纸条
题目传送门
本题其实就是方格取数的那道题,唯二不同的是:
(1) 数据范围变成了\(50\)
(2) 行是\(m\),列是\(n\), 不再是一个正方形,变成了一个矩形,其实都一样。
#include
using namespace std;
const int N = 55;
int n, m;
int w[N][N];
int f[N * 2][N][N];
int main() {
cin >> m >> n;
for (int i = 1; i <= m; i++)
for (int j = 1; j <= n; j++)
cin >> w[i][j];
//左上角是(1,1),k表示两个小朋友所在位置的x+y的和,最多是2*n
for (int k = 2; k <= n + m; k++)
for (int x1 = 1; x1 <= m; x1++)//第一个小朋友竖着走的距离
for (int x2 = 1; x2 <= m; x2++) {//第二个小朋友竖着走的距离
int y1 = k - x1, y2 = k - x2;//计算横着走的距离
//不能出界,只走有效的位置
if (y1 >= 1 && y1 <= n && y2 >= 1 && y2 <= n) {
//PK获取到最优的上一个状态
int t = f[k - 1][x1 - 1][x2];
t = max(t, f[k - 1][x1 - 1][x2 - 1]);
t = max(t, f[k - 1][x1][x2 - 1]);
t = max(t, f[k - 1][x1][x2]);
//将本位置的数值加上
f[k][x1][x2] = t + w[x1][y1];
//如果不是重复的位置,还可以继续加上
if (x1 != x2) f[k][x1][x2] += w[x2][y2];
}
}
//输出DP的结果
printf("%d\n", f[m + n][m][m]);
return 0;
}