Problem Description
ACM亚洲区比赛结束,意味着开始备战明年的浙江省大学生程序设计竞赛了!
杭州电子科技大学ACM集训队也准备开始组队。
教练想把所有的n个队员组成若干支队伍,原则是每支队伍至少一人,最多三人。
现在问题来了:如果已知集训队队员的数量n,请你帮教练计算出所有可能的组队方案有多少种。
特别说明:
队伍没有编号,即如果有A,B,C三人,{A}{BC}与{BC}{A}是同一种组队情况。
Input
输入包含多组测试数据(约1000组),每组数据占一行,包含一个数字n(0<=n<=20),表示ACM集训队的队员人数;n为0,表示输入结束。
Output
请输出n个队员所有可能的组队方案数,每组输出占一行。
Sample Input
1
2
3
4
5
0
Sample Output
1
2
5
14
46
分析:
DP:
定义状态f[i]表示i个人的组合方案。
考虑第i个人,他有三种决策:
1.一个人一个队 那么答案就是 f[i-1];
2.2个人一个队 则需要在前面 i-1个人中再找一个人和他组队,于是C(1,i-1)种选择,对于每一个选择,又有f[i-2]个选择,所以答案就是C(1,i-1)*f[i-2];
3.3个人一个队,原理同上答案是C(2,i-1)*f[i-3];
综上,转移方程为:f[i]=f[i-1]+c[1][i-1]*f[i-2]+c[2][i-1]*f[i-3];
边界:f[1]=1;f[2]=3;f[3]=5;
最后组合数C(m,n)=C(m-1,n-1)+C(m,n-1)
不用死记,原理很捡蛋可以直接推。
对于第n个人来说,他可以是前m个人中的一个:C(m-1,n-1);也可以不是前m个人的一个:C(m,n-1).
一开始没有认真看这道题,,,直接跳过去看别的题了,结果另一道题写T了,,悲伤之余划了一会儿水,最后二十分钟突然发现这是一道简单DPqwq。。。。。。然后开始写,但是悲催的是我不知道怎么算C(m,n)。。。最后用定义法变算边除,当然结果还是wa的还是会爆long long,其实算C的方法也是个简单DP。。。
// Created by ZYD in 2015.
// Copyright (c) 2015 ZYD. All rights reserved.
//
#include <cstdio>
#include <cstdlib>
#include <iostream>
#include <algorithm>
#include <cstring>
#include <climits>
#include <string>
#include <vector>
#include <cmath>
#include <stack>
#include <queue>
#include <set>
#include <map>
using namespace std;
#define Size 100000
#define ll long long
#define mk make_pair
#define pb push_back
#define mem(array) memset(array,0,sizeof(array))
typedef pair<int,int> P;
ll f[50];
ll c[50][50];
int n;
int main()
{
freopen("in.txt","r",stdin);
for(int i=1;i<=20;i++)
{ c[i][i]=1;c[1][i]=i;}
for(int i=3;i<=20;i++)
c[2][i]=c[2][i-1]+c[1][i-1];
// cout<<c[2][3]<<"*";
f[1]=1;f[2]=2;f[3]=5;
for(int i=4;i<=20;i++)
f[i]=f[i-1]+c[1][i-1]*f[i-2]+c[2][i-1]*f[i-3];
// cout<<C(10,20);
// for(int i=1;i<=20;i++)cout<<f[i]<<" ";
while(cin>>n && n!=0)
{
cout<<f[n]<<endl;
}
return 0;
}