PKU_ACM_1088 滑雪

滑雪 
Time Limit:1000MS  Memory Limit:65536K
Total Submit:6256 Accepted:1796 

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 Output


25

 

分析:

用回溯法,思路还是比较简单地,但是会超时,不知道有什么其他的办法

for each 可能的start point:

   从该点回溯分析最长的下降路径

代码如下

PKU_ACM_1088 滑雪 #include  < stdio.h >
PKU_ACM_1088 滑雪#include 
< stdlib.h >
PKU_ACM_1088 滑雪#include 
< string .h >
PKU_ACM_1088 滑雪#include 
< vector >
PKU_ACM_1088 滑雪#include 
< assert.h >
PKU_ACM_1088 滑雪#include 
< math.h >
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪
using   namespace  std;
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪
// #define PI 3.1415926
PKU_ACM_1088 滑雪
// #define DBG 
PKU_ACM_1088 滑雪

PKU_ACM_1088 滑雪
namespace  Baidu09_3
PKU_ACM_1088 滑雪
{
PKU_ACM_1088 滑雪    
struct Row{
PKU_ACM_1088 滑雪        
int data[100];
PKU_ACM_1088 滑雪    }
;
PKU_ACM_1088 滑雪    vector 
<Row> vGraph;
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪    
struct Path{
PKU_ACM_1088 滑雪        
int r;
PKU_ACM_1088 滑雪        
int c;
PKU_ACM_1088 滑雪        
int h;
PKU_ACM_1088 滑雪    }
;
PKU_ACM_1088 滑雪    vector 
<Path>  vPath;
PKU_ACM_1088 滑雪    vector 
<Path>  vPathMax;
PKU_ACM_1088 滑雪    
PKU_ACM_1088 滑雪    
void Test(vector <Row> &vGraph, int R, int C, int r, int c)
PKU_ACM_1088 滑雪    
{
PKU_ACM_1088 滑雪        
int dir[4][2]={
PKU_ACM_1088 滑雪            
{0,1},
PKU_ACM_1088 滑雪            
{0,-1},
PKU_ACM_1088 滑雪            
{1,0},
PKU_ACM_1088 滑雪            
{-1,0},
PKU_ACM_1088 滑雪        }
;
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪        
bool canGo =false;
PKU_ACM_1088 滑雪        
for (int i=0; i<4; i++)
PKU_ACM_1088 滑雪        
{
PKU_ACM_1088 滑雪            
int r1=r+dir[i][0];
PKU_ACM_1088 滑雪            
int c1=c+dir[i][1];
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪            
if(r1>=0 && r1<&& c1>=0 && c1<&& vGraph[r1].data[c1]<vGraph[r].data[c])
PKU_ACM_1088 滑雪            
{
PKU_ACM_1088 滑雪                canGo 
= true;
PKU_ACM_1088 滑雪                Path path 
= {r1, c1, vGraph[r1].data[c1]};
PKU_ACM_1088 滑雪                vPath.push_back(path);
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪                Test(vGraph,R,C,r1,c1);
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪                vPath.pop_back();
PKU_ACM_1088 滑雪            }

PKU_ACM_1088 滑雪        }

PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪        
if(!canGo)
PKU_ACM_1088 滑雪        
{
PKU_ACM_1088 滑雪            
if(vPathMax.size()<vPath.size())
PKU_ACM_1088 滑雪                vPathMax 
= vPath;
PKU_ACM_1088 滑雪            
/*printf("\n----%d\n",vPath.size());
PKU_ACM_1088 滑雪            for(size_t i=0; i<vPath.size(); i++)
PKU_ACM_1088 滑雪                printf("%d ", vPath[i].h);
*/

PKU_ACM_1088 滑雪        }

PKU_ACM_1088 滑雪    }

PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪    
int main()
PKU_ACM_1088 滑雪    
{
PKU_ACM_1088 滑雪        
int R,C;
PKU_ACM_1088 滑雪        scanf(
"%d%d"&R, &C);
PKU_ACM_1088 滑雪        vGraph.reserve(R);
PKU_ACM_1088 滑雪        
PKU_ACM_1088 滑雪        
for(int i=0; i<R; i++)
PKU_ACM_1088 滑雪        
{
PKU_ACM_1088 滑雪            Row row;
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪            
for(int j=0; j<C; j++)
PKU_ACM_1088 滑雪                scanf(
"%d"&row.data[j]);
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪            vGraph.push_back(row);
PKU_ACM_1088 滑雪        }

PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪        
for(int i=0; i<R; i++)
PKU_ACM_1088 滑雪        
{
PKU_ACM_1088 滑雪            
for(int j=0; j<C; j++)
PKU_ACM_1088 滑雪            
{
PKU_ACM_1088 滑雪                vPath.clear();
PKU_ACM_1088 滑雪
PKU_ACM_1088 滑雪                Path path 
= {i, j, vGraph[i].data[j]};
PKU_ACM_1088 滑雪                vPath.push_back(path);
PKU_ACM_1088 滑雪                Test(vGraph,R,C,i,j);
PKU_ACM_1088 滑雪            }

PKU_ACM_1088 滑雪        }

PKU_ACM_1088 滑雪        
PKU_ACM_1088 滑雪        printf(
"%d\n",vPathMax.size());
PKU_ACM_1088 滑雪        
for(size_t i=0; i<vPathMax.size(); i++)
PKU_ACM_1088 滑雪            printf(
"%d ", vPathMax[i].h);
PKU_ACM_1088 滑雪        
return 0;
PKU_ACM_1088 滑雪    }

PKU_ACM_1088 滑雪}
;
PKU_ACM_1088 滑雪

 正确的方法是用动态规划,参考:

 

ACM PKU 1088 滑雪  经典的动态规划备忘录方法(记忆化搜索/Memory function )

http://acm.pku.edu.cn/JudgeOnline/problem?id=1088

非常经典的一道动态规划题,AC的时候心情简直舒畅到了极点.
时间限制是1000MS,如果直接用DFS肯定超时的.
马上想到动归,
用opt[i][j]记录从点node[i][j]出发的最短路径(不算本身,只算延伸;也就是初始值为0)
状态转移方程opt[i][j]=max{ opt[i+1][j],opt[i-1][j],opt[i][j+1],opt[i][j-1] } +1    
也就是说,opt[i][j]的值等于从node[i][j]的上下左右四个方向出发所滑的最长值+1;
而这道题并不是简单的动归,计算opt[i][j]的过程需要类似DFS的递归方法.这就是记忆化搜索. 


Problem Id:1088  User Id:lnmm
Memory:152K  Time:0MS
Language:C++  Result:Accepted

 1 PKU_ACM_1088 滑雪 #include " stdio.h "
 2 PKU_ACM_1088 滑雪 const   int  dx[] = {-1,0,1,0} ,dy[] = {0,1,0,-1} ;
 3 PKU_ACM_1088 滑雪 int  r,c; // r和c分别是行和列
 4 PKU_ACM_1088 滑雪 int  node[ 101 ][ 101 ];  // 放置每个坐标上的高度
 5 PKU_ACM_1088 滑雪 int  opt[ 101 ][ 101 ];  // 放置从每个坐标出发的最优解
 6 PKU_ACM_1088 滑雪
 7 PKU_ACM_1088 滑雪 bool  ok( int  i, int  j)
 8 PKU_ACM_1088 滑雪 {
 9PKU_ACM_1088 滑雪  return (i>=1 && i<=&& j>=1 &&j<=c);
10PKU_ACM_1088 滑雪}

11 PKU_ACM_1088 滑雪
12 PKU_ACM_1088 滑雪
13 PKU_ACM_1088 滑雪
14 PKU_ACM_1088 滑雪 int  dp( int  i, int  j)
15 PKU_ACM_1088 滑雪 {
16PKU_ACM_1088 滑雪    int k;
17PKU_ACM_1088 滑雪    if(opt[i][j]>0return opt[i][j];    //如果已经计算出,直接返回
18PKU_ACM_1088 滑雪    for(k=0;k<4;k++)                    //向四个方向延伸
19PKU_ACM_1088 滑雪    {
20PKU_ACM_1088 滑雪        if(ok(i+dx[k],j+dy[k]))          //如果节点没有超出边界
21PKU_ACM_1088 滑雪            if( node[i+dx[k]][j+dy[k]]<node[i][j] )        //满足滑雪条件
22PKU_ACM_1088 滑雪            {
23PKU_ACM_1088 滑雪                if(  opt[i][j]< dp(i+dx[k],j+dy[k])+1 ) 
24PKU_ACM_1088 滑雪                         opt[i][j]=dp(i+dx[k],j+dy[k])+1;
25PKU_ACM_1088 滑雪            }

26PKU_ACM_1088 滑雪    }

27PKU_ACM_1088 滑雪    return opt[i][j];
28PKU_ACM_1088 滑雪
29PKU_ACM_1088 滑雪
30PKU_ACM_1088 滑雪//       if(ok(i+dx[k],j+dy[k])&&node[i+dx[k]][j+dy[k]]<node[i][j]&&opt[i][j]>dp(i+dx[k],j+dy[k])+1)
31PKU_ACM_1088 滑雪//           opt[i][j]=dp(i+dx[k],j+dy[k])+1;
32PKU_ACM_1088 滑雪
33PKU_ACM_1088 滑雪     
34PKU_ACM_1088 滑雪
35PKU_ACM_1088 滑雪}

36 PKU_ACM_1088 滑雪
37 PKU_ACM_1088 滑雪 void  main()
38 PKU_ACM_1088 滑雪 {
39PKU_ACM_1088 滑雪    int max=0,i,j; 
40PKU_ACM_1088 滑雪    scanf("%d%d",&r,&c);
41PKU_ACM_1088 滑雪
42PKU_ACM_1088 滑雪    for(i=1;i<=r;i++)
43PKU_ACM_1088 滑雪        for(j=1;j<=c;j++)
44PKU_ACM_1088 滑雪            scanf("%d",&node[i][j]);
45PKU_ACM_1088 滑雪   for(i=1;i<=r;i++)
46PKU_ACM_1088 滑雪        for(j=1;j<=c;j++)
47PKU_ACM_1088 滑雪         opt[i][j]=0;
48PKU_ACM_1088 滑雪
49PKU_ACM_1088 滑雪    for(i=1;i<=r;i++)
50PKU_ACM_1088 滑雪        for(j=1;j<=c;j++)
51PKU_ACM_1088 滑雪            if(max<dp(i,j))max=dp(i,j);
52PKU_ACM_1088 滑雪    printf("%d",max+1);  //输出值需要+1 ,因为在前面的计算中,每个点的初始值都是0
53PKU_ACM_1088 滑雪
54PKU_ACM_1088 滑雪    return ;
55PKU_ACM_1088 滑雪}

56 PKU_ACM_1088 滑雪

 http://www.cppblog.com/AClayton/archive/2007/09/17/32336.aspx

你可能感兴趣的:(ACM)