【牛客oj】过桥(逆推dp)

链接:登录—专业IT笔试面试备考平台_牛客网
来源:牛客网
 

题目描述

dd被困在了一个迷幻森林,现在她面前有一条凶险的大河,河中央有nnn个神奇的浮块,浮块按1∼n顺序标号,但两两并不相接,第i个浮块上有一个数字a[i],可能是正数,也可能是负数,每块浮块都附带一个魔法结界用于传送,当a[i]为正数时,dd可以选择传送到第i+k(1≤k≤a[i])个浮块上,当dd抵达n号浮块时才可以顺利脱身,显然不管a[n]是多少,都没有任何意义,当a[i]为负时,dd只能选择标号小于等于i+a[i]的任意一块浮块进行传送,当i+a[i]<1时,默认只能传送到1的位置,每次传送都会花费1s的时间,随着时间的流逝,迷雾森林的空气会被逐渐榨干,她现在在1号浮块,她想知道,她最快多久能顺利脱身,如果始终无法逃脱,请输出−1

输入描述:

第一行一个数n(2≤n≤2000)
接下来一行n个数a[i](1≤|a[i]|≤2000)表示浮块上的数字

输出描述:

输出一行,表示对应的答案

示例1

输入

4
2 2 -1 2

输出

2

说明

1跳到2,1s
2跳到4,1s
共2s 

示例2

输入

2
-1 -2

输出

-1

解题思路:

很明显这是一道dp的题目, 但是如果从头到尾枚举,每次都会具有后效性,不符合dp的逻辑,所以我们逆着遍历,这样可以解决后效性的问题。

倒着枚举每一个节点,如果这个节点的值小于零,就会产生会跳,这样对这个题目其实是没有意义的,可能会有疑问的地方就是如果我跳到了前面的某个节点,这个节点可以到达n节点。但是这样跳到前面的那个节点也需要花费时间1,没有直接从那个节点到终点所花费的时间少,所以这样的做法是没有意义的。如果节点值大于零我就找到最小的花费。如果最终第一个节点的dp值还是无穷大就说明没有路可以走出去。

下面附上ac代码

#include <bits/stdc++.h>
#define INF 0x3f3f3f3f
using namespace std;
int n;
int a[2010];
int dp[2010];   //从1到i所需要的最小花费
int main()
{
    memset(dp,INF,sizeof(dp));
    scanf("%d",&n);
    dp[n]=0;
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&a[i]);
    }
    for(int i=n-1;i>=1;i--)
    {
        if(a[i]<0)    continue;    //产生回跳无意义
        else if(a[i]>0)    //枚举可到达的每个节点,寻找最小值
        {
            for(int j=i+1;j<=i+a[i];j++)
            {
                dp[i]=min(dp[i],dp[j]+1);
            }
        }
    }
    if(dp[1]==INF)  printf("-1\n");
    else printf("%d\n",dp[1]);
    //system("pause");
    return 0;
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值