您的位置:首页 > 其它

CODE[VS] 1039 数的划分

2015-07-14 22:38 375 查看
题目描述 Description

将整数n分成k份,且每份不能为空,任意两种划分方案不能相同(不考虑顺序)。

例如:n=7,k=3,下面三种划分方案被认为是相同的。

1 1 5

1 5 1

5 1 1

问有多少种不同的分法。

输入描述 Input Description

输入:n,k (6<n<=200,2<=k<=6)

输出描述 Output Description

输出:一个整数,即不同的分法。

样例输入 Sample Input

7 3

样例输出 Sample Output

4

数据范围及提示 Data Size & Hint

{四种分法为:1,1,5;1,2,4;1,3,3;2,2,3;}



这是一个划分型DP的题目,对于这个题,我们可以用一个数组dp[i][j]表示把i划分成j份的方案数,那么我们如何写动态转移方程呢:
dp[i][j] = dp[i - j][1] + dp[i - j][2] + ... + dp[i - j][j](i >= j)

好的,来解释一下这个简单的动态转移方程,首先因为要分成j份,所以我们假设在第1份第2份一直到第j份先放上1,保证每一份都有东西。这是第一次,也就是解释了i - j的含义,就是放下这j个1后,还剩下i - j,然后如果剩下的一次都放在第1份,就是dp[i - j][1],如果剩下都放在第一份和第二份就是dp[i - j][2],用递归的思想去思考,很容易得出这个动态转移方程。

那么最后的问题就是如何表示动态转移方程呢。我们考虑i -1, j - 1的情况。
dp[i - 1][j - 1] = dp[i - 1 - j + 1][1] + ... dp[i - 1 - j + 1][j - 1]
所以我们很容易看出来,dp[i - 1][j - 1]和dp[i][j]只相差一个dp[i - j][j],问题便解决了

初始化只要让dp[0][0] = 1即可
下面是代码:
/*************************************************************************
> File Name: 数的划分.cpp
> Author: zhanghaoran
> Mail: chilumanxi@gmail.com
> Created Time: 2015年07月14日 星期二 19时56分03秒
************************************************************************/

#include <iostream>
#include <algorithm>
#include <utility>
#include <cstring>
using namespace std;

int n, k;
int dp[201][7];

int main(void){
cin >> n >> k;
memset(dp, 0, sizeof(dp));
dp[0][0] = 1;
for(int i  = 1; i <= n; i ++){
for(int j = 1; j <= k; j ++){
if(i >= j){
dp[i][j] = dp[i- j][j] + dp[i - 1][j - 1];
}
}
}
cout << dp
[k] << endl;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: