题目链接:戳这里

题意:有A只蚂蚁,来自T个家族,每个家族有ti只蚂蚁。任取n只蚂蚁(S <= n <= B),求能组成几种集合?

这道题可以用dp或母函数求。

多重集组合数也是由多重背包问题拓展出来的一类经典问题,而此类问题也都可以用母函数求.

给大家讲2种方法:

①朴素方法:

状态:dp[i][j]:前i种中选j个可以组成的集合数

决策:第i种选k个,k<=cnt[i] && j-k>=0

转移:dp[i][j]=Σdp[i-1][j-k]

复杂度为O(B*Σant[i])即O(B*A)也即O(A^2),虽说这题A最大可到1e5,但是实际数据水,能过

其实这个所谓的朴素dp算法就是母函数的算法.

附ac代码:

 1 #include <cstdio>
2 #include <cstring>
3 #include <algorithm>
4 using namespace std;
5 typedef unsigned long long ll;
6 const int maxn = 1e3 + 10;
7 const int inf = 0x3f3f3f3f;
8 const int maxx = 1e5 + 10;
9 int dp[2][maxx];
10 int cnt[maxn];
11 int num[maxn];
12 const int mod = 1e6;
13 int main()
14 {
15 int t, a, s ,b, u;
16 scanf("%d %d %d %d", &t, &a, &s, &b);
17 for(int i = 1; i <= a; ++i)
18 {
19 scanf("%d", &u);
20 ++cnt[u];
21 }
22 //处理边界问题,当只有一种蚂蚁的时候,无论多少个蚂蚁,组成的集合都是1
23 for(int i = 0; i <= cnt[1]; ++i) // 这里从0开始赋值是为了后面当j- k = 0时dp[][j-k]=1
24 dp[1][i] = 1;
25 for(int i = 2; i <= t; ++i)
26 {//这里j=0依然是处理边界,比如dp[3][1] = dp[2][0] + dp[2][1]
27 for(int j = 0; j <= b; ++j)
28 {
29 for(int k = 0; k <= min(j, cnt[i]); ++k)
30 {
31 dp[i & 1][j] = (dp[i & 1][j] + dp[(i & 1) ^ 1][j - k]) % mod;
32
33 }
34 // printf("%d %d %d\n", dp[i&1][j], i, j);
35 }
36 memset(dp[(i & 1) ^ 1], 0, sizeof(dp[(i & 1) ^ 1]));
37 }
38 int ans = 0;
39 for(int i = s; i <= b; ++i)
40 {
41 ans = (ans + dp[t & 1][i]) % mod;
42 }
43 printf("%d\n", ans);
44 return 0;
45 }

②优化递推式

状态:dp[i][j]:前i种中选j个可以组成的集合数

决策:第i种不选或者至少选一个

转移: dp[i][j] = dp[i-1][j] + dp[i][j-1] - dp[i-1][j-cnt[i]-1] (j-cnt[i]-1>=0)

优化思路:

根据①可以知道

dp[i][j]=∑{k=0~min(j,cnt[i])} dp[i][j-k]

所以dp[i][j-1]=∑(k=0~min(j-1,cnt[i])} dp[i][j-1-k]

二者之间的关系是:

dp[i][j]=dp[i][j-1]+dp[i][j-1]-dp[i-1][j-cnt[i]-1] 即得出优化的转移方程.

通俗来说,就是从前i种中取j个只与从前i种中取j-1个有两种情况不同,其他都是一样的.

这样就省去了大量的重复运算.

复杂度为O(T*B)

附ac代码:

 1 #include <cstdio>
2 #include <cstring>
3 #include <algorithm>
4 using namespace std;
5 typedef unsigned long long ll;
6 const int maxn = 1e3 + 10;
7 const int inf = 0x3f3f3f3f;
8 const int maxx = 1e5 + 10;
9 int dp[2][maxx];
10 int cnt[maxn];
11 int num[maxn];
12 const int mod = 1e6;
13 int main()
14 {
15 int t, a, s ,b, u;
16 scanf("%d %d %d %d", &t, &a, &s, &b);
17 for(int i = 1; i <= a; ++i)
18 {
19 scanf("%d", &u);
20 ++cnt[u];
21 }
22 //处理边界问题,当只有一种蚂蚁的时候,无论多少个蚂蚁,组成的集合都是1
23 for(int i = 0; i <= cnt[1]; ++i) // 这里从0开始赋值是为了后面当j- k = 0时dp[][j-k]=1
24 dp[1][i] = 1;
25 for(int i = 2; i <= t; ++i)
26 {//这里j=0依然是处理边界,比如dp[3][1] = dp[2][0] + dp[2][1]
27 for(int j = 0; j <= b; ++j)
28 {
29 dp[i & 1][j] = (dp[i & 1][j] + dp[(i & 1) ^ 1][j] + dp[i & 1][j - 1])%mod;
30 if(j - 1 - cnt[i] >= 0)
31 dp[i & 1][j] = (dp[i & 1][j] - dp[(i & 1) ^ 1][j - 1 - cnt[i]] + mod)%mod;//+mod防止负数
32 // printf("%d %d %d\n", dp[i&1][j], i, j);
33 }
34 memset(dp[(i & 1) ^ 1], 0, sizeof(dp[(i & 1) ^ 1]));
35 }
36 int ans = 0;
37 for(int i = s; i <= b; ++i)
38 {
39 ans = (ans + dp[t & 1][i])%mod;
40 }
41 printf("%d\n", ans);
42 return 0;
43 }

参考博客:戳这里

最新文章

  1. ASP.NET 5 RC1 升级 ASP.NET Core 1.0 RC2 记录
  2. MySQL分区表的管理~1
  3. JQurey
  4. Ubuntu MYSQL和Windows MYSQL (非C盘安装)
  5. 硬盘空间满导致mysql ibd文件被删后提示Tablespace is missing for table &#39;db_rsk/XXX&quot;
  6. Linux设备驱动中的并发控制
  7. radioButton
  8. 华为OJ—字符串排序(排序,忽略指定字符排序)
  9. C#检验数据有效性验证类
  10. mysql-no-install 手动安装
  11. NOI2013 树的计数
  12. openstack API debug OpenstackEveryProject_CLI,curl_based
  13. Linux下服务器重启
  14. struts2捕获action类异常
  15. vue element-ui 绑定@keyup事件无效
  16. Non-zero exit code (1)
  17. Let me introduce myself
  18. java时间与js时间
  19. Material Design学习之 ProgreesBar
  20. graphEdit

热门文章

  1. library cache pin解决方法
  2. Eureka详解系列(一)--先谈谈负载均衡器
  3. CodeMonkey少儿编程第3章 times循环
  4. Py-解决粘包现象,tcp实现并发,tcp实现传输文件的程序,校验思路,线程与进程
  5. charles安装使用乱码连手机等问题解决方案
  6. https://learnku.com/docs/go-blog/qihoo/6532 。 heap size went up to 69G, with maximum garbage collection (GC)
  7. SpringMVC听课笔记(九:数据转换 &amp; 数据格式化 &amp; 数据校验)
  8. Pycharm 使用学习
  9. jackson学习之五:JsonInclude注解
  10. Docker综述