小灶第五次作业 dp

编程入门 行业动态 更新时间:2024-10-28 13:21:47

<a href=https://www.elefans.com/category/jswz/34/1715266.html style=小灶第五次作业 dp"/>

小灶第五次作业 dp

题目大意:

  一个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 时候是走到了右下角,题目要求只能向下走或者向左走,于是就说明你不可能绕圈圈,换句话说你只能离目标越来越近,不可能是绕道远的地方再回去,也就省了很多麻烦,横纵坐标之后和步数是有明确的关系的,同时,无论你之前怎么选某一个位置都可以得到,不用担心错过和比较大的。然后就是状态方程: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 ];elsereturn -1;}int ans = dp[step][r1][r2];if(ans != -1)return ans;if(r1 < n && r2 < n)ans = 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] + ((r1==r2)?0: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));}return 0;
}

更多推荐

小灶第五次作业 dp

本文发布于:2024-03-06 11:53:55,感谢您对本站的认可!
本文链接:https://www.elefans.com/category/jswz/34/1715265.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文标签:小灶   作业   第五次   dp

发布评论

评论列表 (有 0 条评论)
草根站长

>www.elefans.com

编程频道|电子爱好者 - 技术资讯及电子产品介绍!