1120: 划分数

Time Limit: 8 Sec  Memory Limit: 128 MB
Submit: 6  Solved: 3
[Submit][Status][Web Board]

Description

有n个无区别的物品,将它们划分成不超过m组,求出划分方法数模M的余数。

限制条件

1 <= m <= n <= 1000

2 <= M <= 10000

Input

每组测试数据输入一行n, m, M。

Output

输出划分数模M的余数。

Sample Input

42 468 6335

Sample Output

2494

HINT

组合数学

Source

可以将问题转变成x1+x2+x3+x4+x5+x6+.....xk+.....xm=n; <=xk;
这类问题一般有这样的限制:每个数的最大值,最小值,每个数是否可以相同.
相当于求不定方程x的非负整数解,但是n等于4时1 2和1 1根据题意,物品是相同的,所以这两种是一样的,
所以不能用校赛那道球放盒子的那种方式,
(考虑第i个盒子,放k个,d[i][j]+=d[i-][j-k];<=k<=m),
从整体进行考虑,对于每一个减去1,由于最小数可能为0,所以有两种情况,一种是最小数为0,一种是最小数为1,
d[m][n]=d[m][n-t]+d[m-][n],最小数为1,减去一,最小数为0,直接去掉0。,初始化d[][]=,
这种方式为什么可以去重呢?
当有多个最小值为0时,由于去0时没有考虑去掉的是哪一个,所以去0操作跟位置没有关系。
(最小数为0,可能有多个,假设是k个,去k次,跟位置无关,去k次就可去完)
假设这个数是0,0,2,不考虑位置的去0,所以可以去重.
将n个数划分为m个不同的正整数解,且最大值小于n
  f1[t][x]=f[t][x-t]+f[t-][x-t]-f[t-][x-(n+)]
将n个数划分为m个不同的正整数解      
  f2[t][x]=f2[t][x-t]+f2[t-][x-t]
下面这两种情况去掉了1 2和1 1类似的重复情况。  
将n个数划分为m个非空集合     
   f3[t][x]=f[t][x-t]+f[t-][x-]
  (最小数为1,去掉一个,如果最小数有多个,由于没有考虑去掉的是哪一个,所以1 , 1是相同的情况)
将n个数划分为m个可为空的集合     
  d[m][n]=d[m][n-m]+d[m-][n]
  (最小数为0,可能有多个,假设是k个,去k次,跟位置无关,去k次就可去完)
   d[m][n]=f3[][n]+f3[][n]+f3[][n]+.....f[k][n]+....f[m][n];
扩展问题:
    有n种物品,第i种物品有a[i]个,不同种类的物品可以互相取分,但相同种类的无法区分,
    从这些物品中取出m个的话,有多少种取法?
解法1
      d[i+][j]
      代表将前i种物品中取出j个,并且每种物品最多去a[i]个,
    d[i+][j]+=d[i][j-k] <=k<=min(j,a[i]);
解法2
    d[i+][j]=d[i+][j-]-d[i][j--a[i]]+d[i][j]
    要求得是从前i个物品中取出j个的方案数,由于此题中1 1 2和1 2 1这两种是不同的,
    所以不用考虑重复,这样就不能给每个数都减去1,我们可以考虑给第i个数减去1,
    a:  当第i个数>=1时,d[i+1][j-1]等于d[i+1][j],然而第i个数的最大值要小于等于a[i],
      再正着看回去,d[i+1][j-1]肯定满足条件,如果给它的第i个数等于a[i],加上1之后,
       d[i+1][j]就不满足条件了,所以需要减去d[i][j-1-a[i]];
    b: 当第i个数等于0时,d[i][j]=d[i+1][j].
#include <iostream>
#include <cstdio>
#include <string.h>
#include <cstdlib>
#include <algorithm>
#include <vector>
using namespace std;
#define maxn 1100
int n,m,MOD;
int d[maxn][maxn];
void init()
{
memset(d,,sizeof(d));
/* for(int i=1;i<=maxn;i++)
{
d[i][1]=1;
d[i][0]=1;
}*/ }
void solve()
{
d[][]=;
for(int i=;i<=m;i++)
for(int j=;j<=n;j++)
{
if(j>=i)
d[i][j]=(d[i][j-i]+d[i-][j])%MOD;
else
d[i][j]=d[i-][j]%MOD;
}
printf("%d\n",d[m][n]);
}
int main()
{
while(~scanf("%d%d%d",&n,&m,&MOD))
{
init();
solve();
}
return ;
}

扩展问题的代码:

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
using namespace std;
#define maxn 1100
int n,m;
int a[maxn];
int dp[maxn][maxn];
int MOD;
void init()
{
// memset(a,0,sizeof(a));
// memset(dp,0,sizeof(dp));
}
void solve()
{
for(int i=;i<=n;i++)
dp[i][]=; for(int i=;i<n;i++)
for(int j=;j<=m;j++)
{
if(j--a[i] >= )
{
dp[i+][j]=(dp[i+][j-] + dp[i][j] -dp[i][j--a[i] ] + MOD) %MOD;
}
else
dp[i+][j]=(dp[i+][j-] + dp[i][j] ) %MOD;
}
for(int i=;i<=n;i++)
{
for(int j=;j<=m;j++)
printf("%d ",dp[i][j]);
printf("\n");
}
}
int main()
{
// freopen("test.txt","r",stdin);
while(~scanf("%d%d%d",&n,&m,&MOD))
{
init();
for(int i=;i<n;i++)
scanf("%d",&a[i]);
solve();
}
return ;
}

snnu1120: 划分数(DP计数问题)的更多相关文章

  1. HDU 4832(DP+计数问题)

    HDU 4832 Chess 思路:把行列的情况分别dp求出来,然后枚举行用几行,竖用几行.然后相乘累加起来就是答案 代码: #include <stdio.h> #include < ...

  2. 划分数 (DP)

    输入: n=4 m=3 M=10000 输出: 4 (1+1+2=1+3=2+2=4) 复杂度(nm) int n,m; int a[MAX]; int dp[MAX][MAX]; //数组 void ...

  3. 5.15 牛客挑战赛40 C 小V和字符串 数位dp 计数问题

    LINK:小V和字符串 容易想到只有1个数相同的 才能有贡献. 知道两个01串 那么容易得到最小步数 大体上就是 第一个串的最前的1和第二个串最前的1进行匹配. 容易想到设f[i][j]表示 前i位1 ...

  4. HDU 4489 The King’s Ups and Downs (DP+数学计数)

    题意:给你n个身高高低不同的士兵.问你把他们按照波浪状排列(高低高或低高低)有多少方法数. 析:这是一个DP题是很明显的,因为你暴力的话,一定会超时,应该在第15个时,就过不去了,所以这是一个DP计数 ...

  5. [HEOI2014]平衡(整数划分数)

    下课了,露露.花花和萱萱在课桌上用正三棱柱教具和尺子摆起了一个“跷跷板”. 这个“跷跷板”的结构是这样的:底部是一个侧面平行于地平面的正三棱柱教具,上面 摆着一个尺子,尺子上摆着若干个相同的橡皮.尺子 ...

  6. 51NOD 1154 回文串的划分(DP)

    思路:参考了网上,思路很清奇,借助vis[i][j]来表示从i到j是否为回文串,回文串这边是用的双重循环来写的:dp[i]用来表示以i结尾的字符串最少的回文串有多长. #include<cstr ...

  7. Day9 - K - Yue Fei's Battle HDU - 5136

    Yue Fei is one of the most famous military general in Chinese history.He led Southern Song army in t ...

  8. Noip模拟42 2021.8.17

    T1 卷 一看跟没有上司的舞会一样,直接敲了然后试个自己造的样例对了就跑了... 然而把它想简单了,乘积取模,还能比大小吗????? 显然不能 所以直接让对数的加和跟着$dp$直接一起跑,比大小的都用 ...

  9. 2021.8.17考试总结[NOIP42]

    $\huge{取模不能比大小!}$ $\huge{取模不能比大小!}$ $\huge{取模不能比大小!}$ 有了打地鼠的前车之鉴,我深信树规板子是可以出现在联赛题里的. 所以T1十分钟码完直接溜了,后 ...

随机推荐

  1. HDU 1278

    题目大意: 从(1,1)到(n,n),每经过一个点都要花费一定的时间,问花最短时间的路径有多少条 dfs+dp 先用bfs把所有到n花费的时间逆向dp计算一遍 再用dfs不断找到前一个对应的较短路径的 ...

  2. SPOJ DCEPC11I

    题目大意: 就是给定一段区间令其中的数增加一个递增序列(也就是说第一个+1,第二个+2.....) 询问操作是区间的和 这里的查询很简单,但是对于添加递增序列入区间就比较搞脑子了 我们需要一个add[ ...

  3. POJ2014 Flow Layout

      Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 3161   Accepted: 2199 Description A f ...

  4. Java jsp页面中jstl标签详解

    JSLT标签库,是日常开发经常使用的,也是众多标签中性能最好的.把常用的内容,放在这里备份一份,随用随查.尽量做到不用查,就可以随手就可以写出来.这算是Java程序员的基本功吧,一定要扎实. JSTL ...

  5. msp430项目编程11

    msp430中项目---步进电机控制系统 1.步进电机工作原理 2.电路原理说明 3.代码(显示部分) 4.代码(功能实现) 5.项目总结 msp430项目编程 msp430入门学习

  6. UVA 11827 Maximum GCD【GCD,stringstream】

    这题没什么好说的,但是输入较特别,为此还WA了一次... 题目链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge& ...

  7. [Bzoj4722]由乃(线段树好题)(倍增处理模数小快速幂)

    4722: 由乃 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 360  Solved: 131[Submit][Status][Discuss] D ...

  8. INFO org.apache.hadoop.ipc.RPC: Server at master/192.168.200.128:9000 not available yet, Zzzzz...

    hadoop 启动时namenode和datanode可以启动,使用jps命令也可以看到进程,但是在浏览器中输入master:50070却没有显示datanode 查看datanode的log日志: ...

  9. wikioi 2147 bitset+map解决

    题目描写叙述 Description 小明是一名天文爱好者,他喜欢晚上看星星.这天,他从淘宝上买下来了一个高级望远镜.他十分开心.于是他晚上去操场上看星星. 不同的星星发出不同的光,他的望远镜能够计算 ...

  10. django 简易博客开发 2 模板和数据查询

    首先还是贴一下项目地址  https://github.com/goodspeedcheng/sblog   因为代码全在上面 上一篇博客我们介绍了 django的安装配置,新建project,新建a ...