滑雪(dp或记忆化搜索)

题意:给你一个二维数组,求最长的递减路线的长度,只能向四个方向延伸。

 

解法1、dp【i】【j】以i、j结尾的最长路线长度。边界:每个数初值为1,

转移:从四周向i、j转移,if(a[i][j]>a[x][y]) dp[i][j] = max(dp[i][j] , dp[x][y]+1);

注意:这里需要按从小到大排序对点进行转移。否则答案有误。

//#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include
#include 
#define ME(x , y) memset(x , y , sizeof(x))
#define SF(n) scanf("%d" , &n)
#define rep(i , n) for(int i = 0 ; i < n ; i ++)
#define INF  0x3f3f3f3f
#define mod 20191117
#define PI acos(-1)
using namespace std;
typedef long long ll ;
int n , m ;
int dir[4][2] = {{1 , 0} , {-1 , 0} , {0 , 1} , {0 , -1}};
int ma ;
int vis[109][109];
int v[109][109];
int dp[109][109];
int b[109][109];

struct node
{
    int x , y  , v;
}a[10009];

bool cmp(struct node a , struct node b)
{
    return a.v < b.v ;
}

int main()
{
    /*#ifdef ONLINE_JUDGE
    #else
        freopen("D:/c++/in.txt", "r", stdin);
        freopen("D:/c++/out.txt", "w", stdout);
    #endif*/
    scanf("%d%d" , &n , &m);
    int l = 0 ;
    for(int i = 0 ; i < n ; i++)//结构体存点值
    {
        for(int j = 0 ; j < m ; j++)
        {
            int x ;
            scanf("%d" , &x);
            b[i][j] = x ;
            a[l].x = i , a[l].y = j ;
            a[l++].v = x ;

        }
    }
    for(int i = 0 ; i < n ; i++)//边界值
    {
        for(int j = 0 ; j < m ; j++)
        {
            dp[i][j] = 1 ;
        }
    }
    sort(a , a + l , cmp);//按值排序
    for(int i = 0 ; i < l ; i++)//按从小到大的点进行转移
    {
        int x = a[i].x ;
        int  y = a[i].y ;
        for(int j = 0 ; j < 4 ; j++)//从四周转移到该点
        {
            int xx = x + dir[j][0];
            int yy = y + dir[j][1];
            if(xx >= 0 && xx < n && yy >= 0 && yy < m && b[xx][yy] < a[i].v)
            {
                dp[x][y] = max(dp[x][y] , dp[xx][yy] + 1);
            }
        }
    }
    int ans = 0 ;
    for(int i = 0 ; i < n ; i++)
    {
        for(int j = 0 ; j < m ; j++)
        {
            ans = max(ans , dp[i][j]);
        }
    }
    cout << ans << endl ;



    return 0;
}

 解法二:记忆化搜索;

//#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include ;
#include 
#include 
#include 
#include 
#include 
#define ME(x , y) memset(x , y , sizeof(x))
#define SF(n) scanf("%d" , &n)
#define rep(i , n) for(int i = 0 ; i < n ; i ++)
#define INF  0x3f3f3f3f
#define mod 10
#define PI acos(-1)
using namespace std;
typedef long long ll ;
int a[109][109];
int dis[109][109];
int dir[4][2] = {{1,0},{-1,0},{0,1},{0,-1}};
int n , m ;
int cnt = 0 ;
int vis[109][109];

int dfs(int x , int y)
{
    if(dis[x][y] != 0){
        return dis[x][y];
    }
    int len =  1 ;//这里是关键。
    for(int i = 0 ; i < 4 ; i++)
    {
        int xx = x + dir[i][0];
        int yy = y + dir[i][1];
        if(xx >= 0 && xx < n && yy >= 0 && yy < m && a[xx][yy] > a[x][y] )
        {
            len = max(len , dfs(xx , yy ) + 1);
        }
    }

    dis[x][y] = len ;//起始点的len是最大的。len表示该点到最长路径的终点的长度
    return len ;//最终返回该点所以路径中的最长路径长度。

}

int main()
{
    scanf("%d%d" , &n , &m);
    for(int i = 0 ; i < n ; i++)
    {
        for(int j = 0 ; j < m ; j++)
        {
            scanf("%d" , &a[i][j]);
        }
    }
    int ma = 0 ;
    for(int i = 0 ; i < n ; i++)
    {
        for(int j = 0 ; j < m ; j++)
        {
            ma = max(ma , dfs(i , j));//返回以所有点为起始点的最长路径。
        }
    }
    cout << ma << endl ;

    return 0;
}

 

你可能感兴趣的:(滑雪(dp或记忆化搜索))