Labyrinth

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 1507    Accepted Submission(s): 520

Problem Description
度度熊是一只喜欢探险的熊,一次偶然落进了一个m*n矩阵的迷宫,该迷宫只能从矩阵左上角第一个方格开始走,只有走到右上角的第一个格子才算走出迷宫,每一次只能走一格,且只能向上向下向右走以前没有走过的格子,每一个格子中都有一些金币(或正或负,有可能遇到强盗拦路抢劫,度度熊身上金币可以为负,需要给强盗写欠条),度度熊刚开始时身上金币数为0,问度度熊走出迷宫时候身上最多有多少金币?
 
Input
输入的第一行是一个整数T(T < 200),表示共有T组数据。
每组数据的第一行输入两个正整数m,n(m<=100,n<=100)。接下来的m行,每行n个整数,分别代表相应格子中能得到金币的数量,每个整数都大于等于-100且小于等于100。
 
Output
对于每组数据,首先需要输出单独一行”Case #?:”,其中问号处应填入当前的数据组数,组数从1开始计算。
每组测试数据输出一行,输出一个整数,代表根据最优的打法,你走到右上角时可以获得的最大金币数目。
 
Sample Input
2
3 4
1 -1 1 0
2 -2 4 2
3 5 1 -90
2 2
1 1
1 1
 
Sample Output
Case #1:
18
Case #2:
4

 
  DP题
  一开始用DFS做的,果断超时。看讨论版才发现需要用DP,当时就蔫了,后来还是查了网上的题解,才明白怎么用DP解这道题,对DP的理解更深了。下面是思路:
  思路:这道题的思路是一列列的确定每一个格子的最大金币数。先确定第一列的最大金币数,因为只能向下走,所以很好求。之后每一列每格的最大金币数可以由前一列求得。
  例如要求dp[i][j](dp[][]的含义是走到这个位置的最大金币数),则从前一列有3种走法到这个格子:
  1、它的左边一个格子,即a[i][j-1],直接向右走一个格子;
  、a[i][j-1]上面的所有格子,可以先向右走一个格,然后向下一直走到a[i][j]的位置;
  、a[i][j-1]下面的所有格子,可以先向右走一个格,然后向上一直走到a[i][j]的位置。
  遍历上述所有路径的过程中需要记录一个最大值,最后这个值就是dp[i][j]的值。
  由此可知,确定该位置的每一种路径,需要遍历m次。而你需要确定每一列的每一个格子的dp[][]值,所以这个算法的时间复杂度为O(n*m*m)
  代码
 #include <iostream>
#include <stdio.h>
using namespace std;
#define inf 0x7fffffff
int a[][];
int dp[][]; //记录走到某一位置的金币最大值
int m,n;
void DP()
{
int i,j,k;
dp[][] = a[][];
for(i=;i<=m;i++) //初始化第一列
dp[i][] = dp[i-][] + a[i][];
for(i=;i<=n;i++){ //每一列 a[][i]
for(j=;j<=m;j++){ //每一列的每一个位置 a[j][i]
//有三种路径到达这个位置。
//1.它左边的位置直接向右走过来。
//2.它左边的位置的上面所有位置向右再一直向下走到这个位置。
//3.它左边的位置的下面所有位置向右再一直向上走到这个位置。
//不断比较,在这个位置处 dp[j][i] 记录以上3条路径的最大值。即为走到这个位置的能拥有的最大金币数。
//这是以 a[j][i] 为重心考虑,遍历到达 a[j][i]的三种路径,求出最大值
int t = dp[j][i-] + a[j][i];
if(t>dp[j][i])
dp[j][i] = t;
for(k=j-;k>=;k--){
//a[k][i-1] --> a[j][i]这个位置所拥有的金币数
t = t-dp[k+][i-]+dp[k][i-]+a[k][i];
if(t>dp[j][i])
dp[j][i] = t;
}
t = dp[j][i-] + a[j][i];
for(k=j+;k<=m;k++){
//a[k][i-1] --> a[j][i]这个位置所拥有的金币数
t = t-dp[k-][i-]+dp[k][i-]+a[k][i];
if(t>dp[j][i])
dp[j][i] = t;
}
}
}
}
int main()
{
int i,j,Case,T;
scanf("%d",&T);
for(Case=;Case<=T;Case++){
scanf("%d%d",&m,&n);
for(i=;i<=m;i++) //输入迷宫,初始化dp[][]
for(j=;j<=n;j++){
scanf("%d",&a[i][j]);
dp[i][j] = -inf;
}
printf("Case #%d:\n",Case);
DP();
printf("%d\n",dp[][n]);
}
return ;
}

  贴上超时的DFS代码

 #include <iostream>
#include <stdio.h>
#include <string.h>
using namespace std;
int Max;
int a[][];
bool isv[][];
int dx[] = {,-,};
int dy[] = {,,};
int m,n;
bool judge(int x,int y)
{
if(x< || x>m || y< || y>n)
return true;
if(isv[x][y])
return true;
return false;
}
void dfs(int x,int y,int money)
{
if(x== && y==n){
if(money>Max)
Max=money;
}
int i;
for(i=;i<;i++){
int nx = x+dx[i];
int ny = y+dy[i];
if(judge(nx,ny))
continue;
//可以走
isv[nx][ny] = true;
dfs(nx,ny,money+a[nx][ny]);
isv[nx][ny] = false;
}
}
int main()
{
int i,j,Case,T;
scanf("%d",&T);
for(Case=;Case<=T;Case++){
scanf("%d%d",&m,&n);
for(i=;i<=m;i++)
for(j=;j<=n;j++)
scanf("%d",&a[i][j]);
printf("Case #%d:\n",Case);
Max=;
memset(isv,,sizeof(isv));
isv[][] = true;
dfs(,,a[][]);
printf("%d\n",Max);
}
return ;
}

2014百度之星资格赛 1004:Labyrinth(DP)的更多相关文章

  1. 2014 百度之星 题解 1004 Labyrinth

    Problem Description 度度熊是一仅仅喜欢探险的熊,一次偶然落进了一个m*n矩阵的迷宫,该迷宫仅仅能从矩阵左上角第一个方格開始走,仅仅有走到右上角的第一个格子才算走出迷宫,每一次仅仅能 ...

  2. 2014百度之星资格赛——Disk Schedule

    2014百度拥有明星格比赛--Disk Schedule Problem Description 有非常多从磁盘读取数据的需求,包含顺序读取.随机读取. 为了提高效率.须要人为安排磁盘读取. 然而.在 ...

  3. 百度之星资格赛 1004 度度熊的午饭时光(01背包+最小序号和+字典序+有bug)

    分析 首先声明一下,我的代码有漏洞的,求大神给个正确代码 思路如下: 首先做一遍01背包记录路径并求出最大总分,令path[i][j]表示第i个物品包含在dp[j]的求值过程中.再逆序枚举money, ...

  4. 2014百度之星资格赛 1001:Energy Conversion(水题,逻辑题)

    Energy Conversion Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  5. ACM学习历程—BestCoder 2015百度之星资格赛1004 放盘子(策略 && 计算几何)

    Problem Description 小度熊喜欢恶作剧.今天他向来访者们提出一个恶俗的游戏.他和来访者们轮流往一个正多边形内放盘子.最后放盘子的是获胜者,会赢得失败者的一个吻.玩了两次以后,小度熊发 ...

  6. 2014百度之星资格赛—— Xor Sum(01字典树)

    Xor Sum Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 132768/132768 K (Java/Others) Total ...

  7. 2017百度之星资格赛 1003:度度熊与邪恶大魔王(DP)

    .navbar-nav > li.active > a { background-image: none; background-color: #058; } .navbar-invers ...

  8. ACM-百度之星资格赛之Labyrinth——hdu4826

    Labyrinth Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total ...

  9. HDU 5686:2016"百度之星" - 资格赛 Problem B

    原文链接:https://www.dreamwings.cn/hdu5686/2645.html Problem B Time Limit: 2000/1000 MS (Java/Others)    ...

随机推荐

  1. Java实现读取文件夹下(包括子目录)所有文件的文件名

    在编程的过程中,经常会用到对文件的读写操作等.比如,找出某一个文件夹下的所有文件名等. 下面的程序给出了,获取某一给定文件夹下所有文件的绝对路径的程序.可以作为某一个模块,在需要的时候直接使用. pa ...

  2. 2016全国研究生数学建模A题多无人机协同任务规划——基于分布式协同多旅行商MTSP遗传算法

    MTSP问题是指:有Ⅳ个城市,要求旅行商到达每个城市各一次,且仅一次,并[旦 1到起点,且要求旅行路线最短.而多旅行商问题M个旅行商从同一个城市(或多个城市)出发.分羽走一条旅路线,且总路程缀短.有关 ...

  3. struts2和hibernate整合的小Demo

    jar包下载地址 创建一个web项目. 导入jar包 配置web.xml <?xml version="1.0" encoding="UTF-8"?> ...

  4. Java语言中几个常用的包

    Java采用包结构来组织和管理类和接口文件.本文介绍Java语言类库中几个常用的包,因为这几个包在软件开发与应用中经常需要用到,其中有些包是必要的.若是离开它,还真不能做事情了. 第一个包:java. ...

  5. ASP.NET 上的 Async/Await 简介

    原文链接 大多数有关 async/await 的在线资源假定您正在开发客户端应用程序,但在服务器上有 async 的位置吗?可以非常肯定地回答“有”.本文是对 ASP.NET 上异步请求的概念性概述, ...

  6. 2016.07.09 offsetWidth 和一个问题。

    javascript 中 offsetWidth 是对象的可见宽度,包滚动条等边线,会随窗口的显示大小改变 clientWidth.offsetWidth.clientHeight区别 IE6.0.F ...

  7. c#.net循环将DataGridView中的数据赋值到Excel中,并设置样式

    Microsoft.Office.Interop.Excel.Application excel =                new Microsoft.Office.Interop.Excel ...

  8. 数位DP题目汇总

    Google Code Jam 2014 Round 1B Problem B hdu 2089 hdu 3555 uestc 250 (原1307) hdu 3652 hdu 3709 Light ...

  9. intel 面试总结

    昨天面了intel的Android部门,基无意外,面试的效果不是很好,很多问题没有回答上来,看来如果要进intel这种大公司光靠SJTU光环是远远不够的,必须要有真才实学.面试的还是一位交大学长,本科 ...

  10. (Win7 x64)NetBeans 8.0.2 使用Tomcat 8作为服务器

    1.下载Apache Tomcat,解压至本地硬盘的根目录. 2.运行CMD,输入: 解压盘符:\apache-tomcat-8.0.xx\bin\service.bat install 3.安装完成 ...