生日聚会 BZOJ 1037

生日聚会

【问题描述】

今天是hidadz小朋友的生日,她邀请了许多朋友来参加她的生日party。 hidadz带着朋友们来到花园中,打算坐成一排玩游戏。为了游戏不至于无聊,就座的方案应满足如下条件:对于任意连续的一段,男孩与女孩的数目之差不超过k。很快,小朋友便找到了一种方案坐了下来开始游戏。hidadz的好朋友Susie发现,这样的就座方案其实是很多的,所以大家很快就找到了一种,那么到底有多少种呢?热爱数学的hidadz和她的朋友们开始思考这个问题…… 假设参加party的人中共有n个男孩与m个女孩,你是否能解答Susie和hidadz的疑问呢?由于这个数目可能很多,他们只想知道这个数目除以12345678的余数。

【输入格式】

仅包含一行共3个整数,分别为男孩数目n,女孩数目m,常数k。

【输出格式】

应包含一行,为题中要求的答案。

【样例输入】

1 2 1

【样例输出】

1

【数据范围】

n , m ≤ 150,k ≤ 20。


题解:

设f[i][j][k][l]为选了i个男生,j个女生,男生与女生最大的差值为k,女生与男生最大的差值为l的方案数

转移方程:(取男生或取女生坐)

 
 1 #include<algorithm>
 2 #include<iostream>
 3 #include<cstring>
 4 #include<cstdlib>
 5 #include<cstdio>
 6 #include<cmath>
 7 using namespace std;
 8 int f[151][151][21][21];
 9 int x, y, z;
10 inline int Max(int x, int y)
11 {
12     return (x < y) ? y : x;
13 }
14 inline void Cal(int &x, int y)
15 {
16     x = (x + y) % 12345678;
17 }
18 int main()
19 {
20     scanf("%d%d%d", &x, &y, &z);
21     f[0][0][0][0] = 1;
22     for(int i = 0; i <= x; ++i)
23         for(int j = 0; j <= y; ++j)
24             for(int k = 0; k <= z; ++k)
25                 for(int l = 0; l <= z; ++l)
26                 {
27                     if(i + 1 <= x && k + 1 <= z) Cal(f[i + 1][j][k + 1][Max(l - 1, 0)], f[i][j][k][l]);
28                     if(j + 1 <= y && l + 1 <= z) Cal(f[i][j + 1][Max(k - 1, 0)][l + 1], f[i][j][k][l]);
29                 }
30     int ans = 0;
31     for(int k = 0; k <= z; ++k)
32         for(int l = 0; l <= z; ++l)
33             Cal(ans, f[x][y][k][l]);
34     printf("%d", ans);
35 }
posted @ 2017-03-02 20:16  草根柴鸡  阅读(151)  评论(0编辑  收藏  举报