EOJ.最短路径

单点时限: 2.0 sec

内存限制: 512 MB

问题描述

输入格式

输出格式

样例:


问题分析:

提示:

要到达A[M-1][N-1],路径上前面一个点只能是A[M-1][N-2]A[M-2][N-1]

可以先计算A[0][0]A[M-1][N-2]A[M-2][N-1]的最短路径,选一个最短的

 记录移动到A[i][j]的最后一次移动方向,通过回溯可以输出整条最短 路径

对每个A[i][j]

存储从A[0][0]A[i][j]的最短路径长度

还需要存储A[0][0]到达A[i][j]的最短路径上最后一次移动的方向

注意点:空间代替时间

考察点:递归


代码解决部分:

#include<stdio.h>
#include<string.h>
#define SIZE 200
typedef struct{int val;char dir}Elem;
void shortestpath(Elem A[][SIZE],int M,int N)   //计算到达每个位置的最短路径长度
{
    int i,j;
    for(i=1;i<M;i++)
    {
        A[i][0].val += A[i-1][0].val;
        A[i][0].dir = 'D';
    }
    for(i=1;i<N;i++)
    {
        A[0][i].val += A[0][i-1].val;
        A[0][i].dir = 'R';
    }
    for(i=1;i<M;i++)
        for(j=1;j<N;j++)
        {
            if(A[i-1][j].val<A[i][j-1].val)         //向下移动
            {
                A[i][j].val += A[i-1][j].val;
                A[i][j].dir = 'D';
            }
            else                                                  //向右移动
            {
                A[i][j].val += A[i][j-1].val;
                A[i][j].dir = 'R';
            }
        }
}

void outpath(Elem A[][SIZE],int r,int c) //递归输出到A[r][c]的最短路径
{
    if(r==0 && c == 0) return;
    if(A[r][c].dir == 'D')
    {
        outpath(A,r-1,c);
        putchar('D');
    }
    else
    {
        outpath(A,r,c-1);
        putchar('R');
    }
}



int main()
{
    Elem A[SIZE][SIZE];
    int M,N;
    int i,j;

    scanf("%d %d",&M,&N);
    for(i=0;i<M;i++)
        for(j=0;j<N;j++)
        {
            scanf("%d",&A[i][j].val);
            A[i][j].dir = 0;
        }
    shortestpath(A,M,N);
    printf("%d\n",A[M-1][N-1].val);
    outpath(A,M-1,N-1);
    putchar('\n');
    return 0;
}


解释:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

NightHacker

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值