核电站

  核电站

Time Limit:1000MS Memory Limit:30000KB

Description 

一个核电站有N个放核物质的坑,坑排列在一条直线上。

如果连续M个坑中放入核物质,则会发生爆炸,于是,在某些坑中可能不放核物质。

任务:对于给定的N和M,求不发生爆炸的放置核物质的方案总数

Input 

该题有多组测试数据,每组数据一行,两个正整数N,M( 1<N≤50,2≤M≤5)

Output 

每组数据只输出一个正整数S,表示方案总数。

Sample Input 

4 3

Sample Output 

13

这题是组合数学的问题,拿到这题要好好分析下,只要能写出递推公式代码就简单了。重点讲下思路,我们用f(x)来表示前x个坑不发生爆炸的方案总数。这题可以分为三种情况:

1、x<m时

当x<m时,说明无论怎么放都不会发生爆炸的情况,那么每个坑可以选择放或不放两张方案,那前x个坑共有f(x)=2^x种放法。也即f(x) = 2*f(x-1)  (这个公式看不明白别着急,待会把第三种情况的公式理解了,这个自然会理解)

2、x=m时,有一种情况不能放,那就是x个坑连在一起的情况,把这种排除即可,则有f(x)=2^x-1种放法。同样可以写成f(x) = 2*f(x-1)-1

3、x>m时,我们用f(x-1)表示前x-1个坑不会发生爆炸的方案总数。那么当到第x个坑时,有两种选择方法即放或者不放,因此放法共有2*f(x-1),但是这个方法包含了一种情况:如果在x个坑选择放了,那么爆炸了这种情况。因此要把这种情况减掉,这种情况数目为f(x-m-1),因此f(x) = 2*f(x-1)-f(x-m-1)。肯定有很多同学对f(x-m-1)这个不明白,下面我来讲解一下:

就拿题目给的n=4,m=3来举个例子:

当到第4个坑时,如果第四个坑选择放,发生爆炸了,那么情况肯定是这样的,2号坑放了,3号坑放了。那么必然1号坑没放。

那只要减掉1号不放这种情况,1号原来有两种选择(放或不放),原来的方案总数为2*f(0),现在明确了就是不放这种情况,那只有1*f(0)种方案,即为f(4-3-1)。                                                                  

现在明白f(x-m-1)了吧。同时也该明白第一种、第二种中公式2*f(i-1)了。                                   现在整个递推公式都出来了,代码很简单,这个题目还有个注意点,就是涉及到组合的话,数可能会非常大,因此int型会溢出把数组f定义为long long即可。下面把代码贴出:

#include <iostream>
<div style="text-align: left;"><strong style="text-align: -webkit-center; color: rgb(0, 0, 153); font-family: 'courier new';font-size:14px; background-color: rgb(255, 255, 255);"></strong></div>#include <cstring>
using namespace std;

int main()
{
    long long f[55];
    int n,m;
    while(cin >> n >> m)
    {
        memset(f,0,sizeof(f));
        f[0] = 1;
        for(int i = 1; i <= n; ++i)
        {
            if(i < m)
            {
                f[i] = 2 * f[i-1];
            }
            else if(i == m)
            {
                f[i] = 2 * f[i-1] -1;
            }
            else
            {
                f[i] = 2 * f[i-1] - f[i-m-1];
            }
        }
        cout << f[n] << endl;
    }
    return 0;
}


                                                                        




你可能感兴趣的:(C++,算法,OJ)