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. hibernate日常BUG总结

    在使用hibernate自动生产表的时候失败, 是配置文件我是从别地方拷贝过来忘记更改,所以报了这个错误. 重新命名了生成表的名称,问题解决! 问题很明显,自动增长的主键应该使用整型. 这里写的是St ...

  2. Dom终

    l创建DOM元素 •createElement(标签名)  创建一个节点 •appendChild(节点)  追加一个节点 –例子:为ul插入li <!DOCTYPE html PUBLIC & ...

  3. css3 transition effect(其它效果)

    http://blog.csdn.net/jerryvon/article/details/8755548 整理了一些其它动画,用的模板为flip模板,只不过CSS3不同 /************* ...

  4. Bitnami --https://bitnami.com/stacks

    Bitnami is an app store for server software. Install your favorite applications in your own servers ...

  5. spinlock原理

    [参考] http://www.searchtb.com/2011/06/spinlock%E5%89%96%E6%9E%90%E4%B8%8E%E6%94%B9%E8%BF%9B.html

  6. ios 使用block中使用self可能产生的循环引用

    在block中调用 self,那么就会引起循环引用问题,那么这是为什么呢? 为什么self会对block进行强引用呢???? 这里推荐一篇关于block的专业文章,http://blog.csdn.n ...

  7. ios NSURLSession completeHandler默认调用quque

    注意 , [[NSURLSession sharedSession] dataTaskWithRequest:request completionHandler:^(NSData *data, NSU ...

  8. 士兵杀敌(三)_RMQ(区间最值查询)

    士兵杀敌(三) 时间限制:2000 ms  |  内存限制:65535 KB 难度:5   描述 南将军统率着N个士兵,士兵分别编号为1~N,南将军经常爱拿某一段编号内杀敌数最高的人与杀敌数最低的人进 ...

  9. 从下拉菜单拖拽一个元素 出来,插入到页面中的app 列表中

    1,实现功能:从下拉菜单拖拽一个元素 出来,插入到页面中的app 列表中 并实现app向后移动一个元素的位置: 2.实现思路: 01.遍历下拉菜单,添加拖拽方法,实现位置移动功能: 02.遍历app列 ...

  10. LINUX查看系统日志

    cat tail -f 日 志 文 件 说    明 /var/log/message 系统启动后的信息和错误日志,是Red Hat Linux中最常用的日志之一 /var/log/secure 与安 ...