题目大意:
一个n*m的格子,Baker Vai要从(1,1)到(n,m)再回到(1,1),每到一个格子可以收集格子上的数字(每个格子只能走一次,(1,1)这个格子除外),问最终搜集的数字之和最大为多少?
解题思路:
可以把题目转化为求两个对象同时从(1,1)出发到(n,m)途中不能相遇,状态转移的时候可以用dp[step][x][y],step代表当前步数,x,y分别代表两者当前所在的行(所在列可以直接求出来)。显然step = m+n-1 时候是走到了右下角,题目要求只能向下走或者向左走,于是就说明你不可能绕圈圈,换句话说你只能离目标越来越近,不可能是绕道远的地方再回去,也就省了很多麻烦,横纵坐标之后和步数是有明确的关系的,同时,无论你之前怎么选某一个位置都可以得到,不用担心错过和比较大的,而且两个人要走到同一位置也肯定是同时到达的。如果走到了同一个,直接返回0,(后续比较肯定不要这个,走的不同最多也是救不了人),然后就是状态方程:ans = max(两个都向下,一个向右一个向下,一个向下一个向右,都向右)注意,记忆化搜索,如果以前搜到这里过,直接返回,每一层算最后的ans加上的是本层的两个人数。
#include <iostream> #include <cstdio> #include <cstring> using namespace std; int a[110][110],dp[220][110][110]; int n, m; int dfs(int step,int r1,int r2) { if(step == n+m-1) { if(r1 == r2 && r1 == n && step-r1+1 == m) return a[r1][step-r1+1 ]; else return -1; } if(r1==r2 && step != 1) return 0; if( dp[step][r1][r2] != -1) return dp[step][r1][r2]; int ans = 0; if(r1 < n && r2 < n) ans =max(dp[step][r1][r2], dfs(step+1,r1+1,r2+1)); if(r1 < n && step-r2+1 < m) ans = max(ans,dfs(step+1,r1+1,r2)); if(r2 < n && step-r1+1 < m) ans = max(ans,dfs(step+1,r1,r2+1)); if(step-r1+1 < m&&step-r2+1 < m) ans = max(ans,dfs(step+1,r1,r2)); ans += a[r1][step-r1+1] + a[r2][step-r2+1]; dp[step][r1][r2] = ans; return ans; } int main() { int T; scanf("%d",&T); for (int i = 1 ; i <= T ; i++) { scanf("%d%d",&n,&m); for (int j = 1 ; j <= n ; j++) for (int k = 1 ; k <= m ; k++) scanf("%d",&a[j][k]); memset(dp,-1,sizeof(dp)); printf("Case %d: %d\n",i,dfs(1,1,1)-a[1][1]); } return 0; }wa了好久的原因是考虑到第一次r1 == r2是一定的,所以我写的r1 != 1,这样就出现如果两个人同时走到第一行的另一个位置,就出现了问题,所以要变成step!=1,希望大家都1A