链接:登录—专业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;
}