[ACM] POJ 1664 放苹果(n个相同小球放入m个相同盒子)

放苹果
Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 25952   Accepted: 16509

Description

把M个同样的苹果放在N个同样的盘子里,允许有的盘子空着不放,问共有多少种不同的分法?(用K表示)5,1,1和1,5,1 是同一种分法。

Input

第一行是测试数据的数目t(0 <= t <= 20)。以下每行均包含二个整数M和N,以空格分开。1<=M,N<=10。

Output

对输入的每组数据M和N,用一行输出相应的K。

Sample Input

1
7 3

Sample Output

8

Source


将n个相同的小球放入m个相同的盒子,问有多少种方法。

 解析:有些题目可以转化为这样的题目,比如有n个相同的小球,m个相同的盒子,要求每个盒子里面至少有k个小球,问一共有多少种方法:http://blog.csdn.net/duanxian0621/article/details/7864791  方法为:预先在每个盒子里面已经放好k个小球,那么还剩下n-k*m 个小球,然后就转化为了,把n-k*m个相同的小球放入m个相同的盒子里面,有多少种方法。

 

分情况讨论:a[i][j]表示i个小球放入j个盒子的方法数

①    当放入小球后,球数最少的盒子为空,那么就相当于,把i个小球放入j-1个盒子里面,即a[i][j-1]。

②    当放入小球后,球数最少的盒子不为空,那么说明每个盒子里面都至少有一个小球,那么就预先把每个盒子里面放入一个小球好了,有 a[i-j] [j]。

所以 ,递推公式为 a[i][j]= a[i][j-1]+a[i-j][j],这里是i>=j的。

当i<j的时候,无论怎么放,肯定会存在空盒,不妨先拿走一个空盒(保证有空盒),然后把i个相同的小球放入剩余的j-1个相同的盒子中,有a[i][j-1]种方法,所以a[i][j]=a[i][j-1].

 

综上:将n个相同的小球放入m个相同的盒子有

当i>=j时,a[i][j]=a[i][j-1]+a[i-j][j];

当i<时,a[i][j]=a[i][j-1],(仔细想想,这里也可以写成a[i][j]=a[i][i],因为a[i][i]=a[i][i+1]=a[i][i+2]=a[i][i+3]……..

初始化a数组代码为:

int a[maxn][maxn];

void prepare()

{

   for (int i=0;i<m;i++)

   a[i][1]=a[0][i]=1;

   for (int i=1;i<maxn;i++)

       for (int j=2;j<maxn;j++)

           if (j<=i)

               a[i][j]=(a[i-j][j]+a[i][j-1])%MOD;

           else

               a[i][j]=a[i][i]; //其实是a[i][j-1],不过和a[i][i]是一样的。

}

代码:

#include <iostream>
using namespace std;
const int maxn=11;
int a[maxn][maxn];

void init()
{
    for(int i=1;i<=10;i++)
        a[0][i]=1,a[i][1]=1;
    for(int i=1;i<maxn;i++)
        for(int j=2;j<=maxn;j++)
            if(j<=i)
                a[i][j]=a[i][j-1]+a[i-j][j];
            else
                a[i][j]=a[i][i];
}

int main()
{
    init();
    int t,n,m;
    cin>>t;
    while(t--)
    {
        cin>>n>>m;
        cout<<a[n][m]<<endl;
    }
    return 0;
}


[ACM] POJ 1664 放苹果(n个相同小球放入m个相同盒子),,5-wow.com

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。