poj1088 经典DP

滑雪

Time Limit: 1000MS   Memory Limit: 65536K
Total Submissions: 88296   Accepted: 33100

Description

Michael喜欢滑雪百这并不奇怪, 因为滑雪的确很刺激。可是为了获得速度,滑的区域必须向下倾斜,而且当你滑到坡底,你不得不再次走上坡或者等待升降机来载你。Michael想知道载一个区域中最长底滑坡。区域由一个二维数组给出。数组的每个数字代表点的高度。下面是一个例子

 1  2  3  4 5
16 17 18 19 6
15 24 25 20 7
14 23 22 21 8
13 12 11 10 9

一个人可以从某个点滑向上下左右相邻四个点之一,当且仅当高度减小。在上面的例子中,一条可滑行的滑坡为24-17-16-1。当然25-24-23-...-3-2-1更长。事实上,这是最长的一条。

Input

输入的第一行表示区域的行数R和列数C(1 <= R,C <= 100)。下面是R行,每行有C个整数,代表高度h,0<=h<=10000。

Output

输出最长区域的长度。

Sample Input

5 5
1 2 3 4 5
16 17 18 19 6
15 24 25 20 7
14 23 22 21 8
13 12 11 10 9

Sample Outpu25
题目大意:汉语的,很好懂,就是让你求最长的递减路径有多长,注意是求下降的次数而不是下降的距离。
思路分析:第一感觉dfs可以做,但是做该题之前发现这道题是归到DP里面的,正好刚学习了一些DP的知识,这道题也算是正式做的第一道DP题目吧,
首先要确定状态,在本题中每一点的状态就是以该点为最高点可以下降的最长路径,然后要明确一状态是如何进行转移的,以数组f来记录每一点的状态,

避免重复搜索,这一点要优于DFS,简单分析我们就可以找到递推关系,f[i][j]=max{f[i-1][j],f[i][j-1],f[i][j+1],f[i+1][j]}+1,代码的实现就要容易

的多了,但是弱在写dp函数判定是否状态转移的时候把行和列搞反了,wa了几发,做题一定要谨慎啊!

代码:

/*dp,记忆化搜索
首先应该确定状态
用数组来保存每一点的状态,
状态的转移在dp函数中已经给出
*/
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#include <stack>
using namespace std;
const int maxn=110;
int m[maxn][maxn],f[maxn][maxn];//f数组代表以该点为最高点的最多的步数
int R,C;
int dp(int x,int y)
{
    if(f[x][y]) return f[x][y];//避免重复搜索
    f[x][y]=1;//每一点下滑梯数最小为1
    if(x>=1&&m[x][y]>m[x-1][y]) f[x][y]=max(f[x][y],dp(x-1,y)+1);
    if(y>=1&&m[x][y]>m[x][y-1]) f[x][y]=max(f[x][y],dp(x,y-1)+1);
    if(x<R-1&&m[x][y]>m[x+1][y]) f[x][y]=max(f[x][y],dp(x+1,y)+1);
    if(y<C-1&&m[x][y]>m[x][y+1]) f[x][y]=max(f[x][y],dp(x,y+1)+1);
    return f[x][y];
}
int main()
{
    int i,j;
    scanf("%d%d",&R,&C);
        memset(f,0,sizeof(f));
        for(i=0;i<R;i++)
            for(j=0;j<C;j++)
            cin>>m[i][j];
        for(i=0;i<R;i++)
            for(j=0;j<C;j++)
            dp(i,j);
      int t=f[0][0];
      for(i=0;i<R;i++)
        for(j=0;j<C;j++)
           if(t<f[i][j]) t=f[i][j];
           cout<<t<<endl;
    return 0;
}

时间: 2024-10-12 11:49:22

poj1088 经典DP的相关文章

51nod 1412 AVL树的种类(经典dp)

http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1412 题意: 思路: 经典dp!!!可惜我想不到!! $dp[i][k]$表示i个结点,最大深度为k的形态数. 它的转移方程就是: dp[i][k] += dp[i - 1 - j][k - 1] * dp[j][k - 1] dp[i][k] += 2 * dp[i - 1 - j][k - 2] * dp[j][k - 1] j是右子树结点个数,如果除去根结点,是不

NYOJ 16 矩形嵌套(经典DP)

http://acm.nyist.net/JudgeOnline/problem.php?pid=16 矩形嵌套 时间限制:3000 ms  |           内存限制:65535 KB 难度:4 描述 有n个矩形,每个矩形可以用a,b来描述,表示长和宽.矩形X(a,b)可以嵌套在矩形Y(c,d)中当且仅当a<c,b<d或者b<c,a<d(相当于旋转X90度).例如(1,5)可以嵌套在(6,2)内,但不能嵌套在(3,4)中.你的任务是选出尽可能多的矩形排成一行,使得除最后一个

POJ 1160:Post Office 邮局经典DP

Post Office Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 17168   Accepted: 9270 Description There is a straight highway with villages alongside the highway. The highway is represented as an integer axis, and the position of each villa

最长上升子序列--经典dp

最长上升子序列 Time Limit: 3000ms   Memory limit: 65536K  有疑问?点这里^_^ 题目描述 一个数的序列bi,当b1 < b2 < ... < bS的时候,我们称这个序列是上升的.对于给定的一个序列(a1, a2, ..., aN),我们可以得到一些上升的子序列(ai1, ai2, ..., aiK),这里1<= i1 < i2 < ... < iK <= N.比如,对于序列(1, 7, 3, 5, 9, 4, 8)

pku 1160 Post Office 四边形不等式优化 经典DP

pku 1160 Post Office 四边形不等式优化 经典DP 邮局 经典的动态规划问题,主要体现在状态的设计和可以用四边形不等式优化上 题意是:给你n个村庄,然后让你用m个邮局对这些村庄进行覆盖,然后让你设计覆盖方式使得每个村庄到其对应邮局的路程和最短 本题状态的设计的灵感来源于"覆盖"这个点,最优子结构其实就是用 m 个邮局覆盖,以及用 m-1个邮局覆盖 那么,状态为dp[n][m] 为前 n 个村庄用 m 个邮局进行覆盖使得每个村庄到其对应的邮局的最短路程和 转移方程:dp

soj 1033 City Road_经典dp

题目链接 题意:给你一个n*m的图,给你b个矩形(在图中,不能通过),只能向上和右走,问左下角到右上角有多少种走法. 思路:经典的dp,dp[i][j]=max(dp[i-1][j],dp[i][j-1]),但是n*m实在太大,不可能直接开这么大的数组,想一下只需要两行的数据就能求出最优解,所以用滚动数组. dp[k][j]=max(dp[1-k][j],dp[k][j-1]),k代表当前行,1-k代表前一行. #include <iostream> #include<cstdio>

Codeforces 176B 经典DP

非常好的一个题目,CF上的DP都比较经典 题意就是 给定一个串A,B,正好执行K次操作,每次操作可以把 A串从中间切开,并调换两部分的位置,问最后得到B串共有多少种不同的切法(只要中间有一次不同,即视为不同) 首先,题目的一个关键点一定要抓到,就是 ,不管怎么切 然后调换位置,其实串根本没变,你把串想成一个环,从某一点分成两部分并且交换位置,其实就是把串的起点变到了该点,这是很关键也是最机智的一点 然后,我们要发现规律,你纸上模拟也行,推理也行.. 我们发现:1.首先原串(即以0号字母开头的)个

poj1161Post Office【经典dp】

题目:poj1161Post Office点击打开链接 题意:给出一条直线上的n个坐标表示村庄的位置,然后要在上面建p个邮局,村民优先选择去近的邮局,问所有村庄去邮局的最小距离和是多少? 分类:区间dp 分析:对于任意一个村庄,只有两种选择,要么在这儿建邮局,要么不建,我们可以预处理出来任意两件建立一个邮局的的最小距离w[i][j],而对于任意两点,建立一个邮局的最优方案是建立在两点的中位数上,即(i+j)/2,位置. 对于任意两点 i---j ,建立两个邮局的最优结果我们可以由建立一个的得到,

NYOJ - 矩形嵌套(经典dp)

矩形嵌套时间限制:3000 ms | 内存限制:65535 KB 描述 有n个矩形,每个矩形可以用a,b来描述,表示长和宽.矩形X(a,b)可以嵌套在矩形Y(c,d)中当且仅当a<c,b<d或者b<c,a<d(相当于旋转X90度).例如(1,5)可以嵌套在(6,2)内,但不能嵌套在(3,4)中.你的任务是选出尽可能多的矩形排成一行,使得除最后一个外,每一个矩形都可以嵌套在下一个矩形内. 输入第一行是一个正正数N(0<N<10),表示测试数据组数,每组测试数据的第一行是一