[POJ1664] 放苹果 (动态规划,组合数学)

题目描述

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

输入输出格式

输入格式:

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

输出格式:

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

输入输出样例

输入样例#1:

1
7 3

输出样例#1:

8

输入样例#2:

3
3 2
4 3
2 7

输出样例#2:

2
4
2

Solution

这道题是我做 P1655 的时候看到的,似乎又是一道DP来解组合数学.

状态很好定义:

\[f[i][j] \]

表示当前有 i 个苹果,放了 j 个箱子的方案数.

初始化:
1)只有当没箱子或者没苹果的时候我们才赋成0
2)当没有苹果,只有一只箱子的时候,我们赋为1,此时只有一种情况.就是不放


前导状态
1)可以有多个箱子不放,此处通过递归来实现
2)全部都放,所以需要m个苹果来每一层至少垫上一个,然后又剩下n-m个苹果随意放

转移方程:

\[f[i][j]=f[i][j-1]+f[i-j][j] \]

代码

#include<bits/stdc++.h>
using namespace std;
int t;

int dp(int n,int m)
{
    if(n<0||m<1)return 0;
    if(n==0&&m==1)return 1;
    return dp(n,m-1)+dp(n-m,m);
}

int main()
{
    ios::sync_with_stdio(false);
    cin>>t;
    while(t--)
    {
        int n,m;
        cin>>n>>m;
        cout<<dp(n,m)<<endl;
    }
}

...






当然了,作为一道组合类的题目,也可以直接作为一道生成函数的模板题来做.但是我似乎没打,只给出一个网址,日后会补生成函数的.
https://blog.csdn.net/dlutjwh/article/details/69217270

posted @ 2018-05-30 22:02  Kevin_naticl  阅读(286)  评论(0编辑  收藏  举报