NOIP 2005 Senior Problem 2 - 过河 (状态压缩DP)

题目描述
在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧。在桥上有一些石子,青蛙很讨厌踩在这些石子上。由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙可能到达的点看成数轴上的一串整点:0,1,……,L(其中 L 是桥的长度)。坐标为 0的点表示桥的起点,坐标为 L 的点表示桥的终点。青蛙从桥的起点开始,不停的向终点方向跳跃。一次跳跃的距离是 S 到 T 之间的任意正整数(包括 S 和 T )。当青蛙跳到或跳过坐标为 L 的点时,就算青蛙已经跳出了独木桥。
题目给出独木桥的长度 L ,青蛙跳跃的距离范围 S , T ,桥上石子的位置。你的任务是确定青蛙要想过河,最少需要踩到的石子数。
输入输出
输入的第一行有一个正整数 L(1 ≤ L ≤ 10^9),表示独木桥的长度。第二行有三个正整数S , T , M,分别表示青蛙一次跳跃的最小距离,最大距离,及桥上石子的个数,其中1 ≤ S ≤ T ≤ 10,1 ≤ M ≤ 100。第三行有 M 个不同的正整数分别表示这 M 个石子在数轴上的位置(数据保证桥的起点和终点处没有石子)。所有相邻的整数之间用一个空格隔开。
输出只包括一个整数,表示青蛙过河最少需要踩到的石子数。
对于 30% 的数据,L ≤ 10000;对于全部的数据,L ≤ 10^9。
——————分割线—————–
一开始我想到了搜索,但是:搜索桥有困难(桥的长度10^9)
搜索石子更困难(石头的分布是没有任何规律)
加上不好想,直接PASS掉,然后很容易想到了DP。。。
DP的状态转移方程很好推:
设dp[i]为青蛙到达 i 位置最少需要踩到的石子数;stone[i] 表示 i 位置上是否有石头,1为有,0为没有;j 代表的是青蛙跳到 i 位置时的可能的区间范围。
则有:dp[i] = min { dp[i-j] + stone [i] } ( j∈[S , T] )
简化为:dp[i] = min { dp[i-j] }+ stone [i] ( j∈[S , T] )
RT
设第 k 个石子座标为 x ,第 k –1 个石子和第 k 个石子间距离足够大,则青蛙从两个石子间跳到第 k 个石子及之后的位置有:x 、x+1、x+2、x+3……x+t –1。如果我们能保证,将石子 k –1 和石子 k 之间的距离缩短(即减少状态)后,青蛙依然能跳到这些位置,则可以平移。而这一点我们可以通过在两个石子间保留 1 个最小公倍数单位长度得到保证。
借鉴了多位大神的想法,代码:

#include <bits/stdc++.h> 
#define MN 105
using namespace std;
int L,s,t,m,ans;
int a[MN];  //保存石子位置 
int f[MN*100];  //f[x]表示青蛙跳到位置i最少踏的石子数 
bool stone[MN*100]; //stone[x]表示位置x是否是石子,0表示不是,1表示是 
int main()
{
    memset(a,0,sizeof(a));
    memset(f,0,sizeof(f));
    memset(stone,0,sizeof(stone));
    scanf("%d%d%d%d",&L,&s,&t,&m);
    ans=0;
    a[m+1]=L; //向桥的最后一个位置放置石头,方便后边计算 
    for (int i=1;i<=m;i++) scanf("%d",&a[i]);
    sort(a+1,a+m+1);  //对桥中间石子位置排序,这步必须要有,没有就WA 
    if(s==t){  //情况特判:这种情况只需考查石子是否是石子的倍数即可 
        for(int i=1;i<=m;i++) if(a[i]%s==0) ans++;
        cout<<ans<<endl;   
    }
    else{
        int d(0),k=s*t,x;  //d表示累加平移量,k表示s和t的公倍数 
        for(int i=1;i<=m+1;i++) 
        {
            x=a[i]-d-a[i-1];  //x表示第i个石子和第i-1个石子的距离 
            if (x>k) d+=x-k;  //超过公倍数部分用作平移 
            a[i]=a[i]-d;
            stone[a[i]]=1;  //标记平移后位置是石子 
        }
        stone[a[m+1]]=0; //RESET,桥尾不是石子  
        for(int i=1;i<=a[m+1]+t-1;i++)  //考查桥上到桥尾的所有位置 
        {
            f[i]=105;   
            for (int j=s;j<=t;j++) //在i的前一个位置中找一个经历石子最少的 
               if (i>=j) f[i]=min(f[i],f[i-j]); //判断i>=j否则无意义 
            f[i]+=stone[i];  //加上当前位置石子数 
        }
        ans=MN;
        for (int i=a[m+1];i<=a[m+1]+t-1;i++)  //在跳过桥后所有位置中找一个最小值 
            if(f[i]<ans) ans=f[i];
        cout<<ans<<endl;  
    }
    return 0;
}
  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值