51nod 1597 有限背包计数问题 dp
来源:互联网 发布:mac怎么恢复废纸篓 编辑:程序博客网 时间:2024/06/06 03:02
题意
你有一个大小为n的背包,你有n种物品,第i种物品的大小为i,且有i个,求装满这个背包的方案数有多少
两种方案不同当且仅当存在至少一个数i满足第i种物品使用的数量不同
1<=n<=10^5,答案对23333333取模
分析
设lim=sqrt(n),我们把所有物品按照大于lim和不大于lim分成两部分。对于大于lim的部分,显然每种物品是一定用不完的,且在这部分所选物品数量一定不会超过lim。
那么我们可以对大于lim的部分进行dp,设g[i,j]表示我选了i种物品,和为j的方案数。转移为g[i,j]=g[i-1,j-lim-1]+g[i,j-i]。
再对不大于lim的部分进行dp,设f[i,j]表示前i种物品中选了若干种其和为j的方案数。转移的时候可以用类似单调队列优化多重背包的方法来优化。
最后统计答案即可,
时间复杂度
代码
#include<iostream>#include<cstdio>#include<cstdlib>#include<cstring>#include<algorithm>#include<cmath>using namespace std;typedef long long LL;const int N=100005;const int MOD=23333333;int n,lim,f[2][N],g[2][N],s[N];int main(){ scanf("%d",&n); lim=sqrt(n); g[0][0]=1;s[0]++; int now=0; for (int j=1;j<=lim;j++) { now=1-now; memset(g[now],0,sizeof(g[now])); for (int i=lim+1;i<=n;i++) g[now][i]=(g[now][i-j]+g[now^1][i-lim-1])%MOD,(s[i]+=g[now][i])%=MOD; } f[0][0]=1; now=0; for (int i=1;i<=lim;i++) { now=1-now; for (int j=0;j<i;j++) { int s=0; for (int k=j;k<=n;k+=i) { (s+=f[now^1][k])%=MOD; if (k-i*i-i>=0) s=(s-f[now^1][k-i*i-i]+MOD)%MOD; f[now][k]=s; } } } int ans=0; for (int i=0;i<=n;i++) (ans+=(LL)f[now][i]*s[n-i]%MOD)%=MOD; printf("%d",ans); return 0;}
阅读全文
0 0
- [背包DP || 多项式] 51Nod 1597 有限背包计数问题
- 51nod 1597 有限背包计数问题 dp
- [DP] 51Nod 1597 有限背包计数问题
- [DP]51 Nod 1597——有限背包计数问题
- 51nod 1597 有限背包计数问题[dp][阈值]
- 51nod 1597 有限背包计数问题
- 51Nod-1597-有限背包计数问题
- [背包DP][小技巧] LOJ#6089. 小 Y 的背包计数问题 && 51NOD 1597 有限背包计数问题
- 【背包+阈值优化】51Nod 1597 有限背包计数问题
- 【51nod1597】【DP】有限背包计数问题
- 51nod 1201[整数划分] 1259[整数划分V2] 1597 [有限背包计数问题]
- 1597 有限背包计数问题
- 有限背包计数问题 (分类dp)
- 有限背包计数问题
- 51nod1597 有限背包计数问题
- [51nod1597] 有限背包计数问题
- 51nod1597 有限背包计数问题[DP][分类讨论][前缀和]
- 【阈值优化+背包】51Nod1597[有限背包计数问题]题解
- lintcode 173 单链表插入排序 python
- 用CSS3美化半个字符巧妙方法
- JavaScript中apply()、call()和bind()的相似与区别
- 根据函数名称导出函数
- sort排序
- 51nod 1597 有限背包计数问题 dp
- MongoDB的使用
- Lucene第三天入门
- 基于sklearn的文本特征提取与分类
- 3.1 Spark-RDD算子分类
- memcpy、strcpy和strncpy的简介
- 去耦、旁路、滤波电容的区别
- 我的人生路-离开大学后2
- codeforces Div.2 #429 B.Godsend C. Leha and Function