核电站
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
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;
}