[ABC221H] Count Multiset

题面翻译

输入两个正整数 \(N,M\),并存在一个集合,问你一个长度为 \(k\) 的合法集合存在多少个?你需要回答 \(k\) 的值为 \(1\) 到 \(N\) 的每种情况。

一个合法的集合定义指长度为 \(k\),元素和为 \(N\),每一个数字在集合中出现的次数都小于等于 \(M\) 的集合。

题目描述

正の整数 $ N $, $ M $ が与えられます。

$ k=1,2,\ldots,N $ について以下の値を求め、$ 998244353 $ で割ったあまりをそれぞれ出力してください。

  • $ k $ 個の正整数からなる多重集合 $ A $ のうち、以下の $ 2 $ つの条件をすべて満たすものの個数

    • $ A $ に含まれる要素の総和は $ N $
    • 任意の正整数 $ x $ について、$ A $ は $ x $ を高々 $ M $ 個しか含まない

输入格式

入力は以下の形式で標準入力から与えられる。

$ N $ $ M $

输出格式

$ N $ 行に渡って出力せよ。$ i\ (1\ \leq\ i\ \leq\ N) $ 行目には、$ k=i $ の場合の答えを出力すること。

样例 #1

样例输入 #1

4 2

样例输出 #1

1
2
1
0

样例 #2

样例输入 #2

7 7

样例输出 #2

1
3
4
3
2
1
1

提示

制約

  • $ 1\ \leq\ M\ \leq\ N\ \leq\ 5000 $
  • 入力はすべて整数

Sample Explanation 1

- $ k=1 $ のとき、問題文中の条件を満たすような多重集合 $ A $ は $ {4} $ の $ 1 $ 通りです。 - $ k=2 $ のとき、問題文中の条件を満たすような多重集合 $ A $ は $ {1,3} $ と $ {2,2} $ の $ 2 $ 通りです。 - $ k=3 $ のとき、問題文中の条件を満たすような多重集合 $ A $ は $ {1,1,2} $ の $ 1 $ 通りです。 - $ k=4 $ のとき、問題文中の条件を満たすような多重集合 $ A $ は $ 1 $ つも存在しません。

解析

一道神奇的 dp 题。有个神奇的转移方式。

看到题目直接联想到背包,但发现 相同元素不超过 \(m\) 这个限制条件很毒瘤。无法用背包转移。然后死活想不出来其他方法。

根据题解,我们不按照背包的思路想,就把他看成一段序列,元素为非负整数(如果有负数,那么每一种情况都有无穷种方案)。令 \(dp[i][j]\) 表示集合里有 \(i\) 个元素,和为 \(j\) 的方案数。考虑在整个序列里添加 \(0\),那么它只对 \(i\) 有贡献,可以得到:\(dp[i][j]+=\sum\limits_{k=1}^{m}dp[i-k][j]\)。但只有 0 不够,考虑在序列里的每一个数加 \(1\),那么它只对 \(j\) 有贡献,可以得到:\(dp[i][j]+=dp[i][j-i]\)。

但还有个 相同元素不超过 \(m\) 这个限制条件,我们不知道一个 \(dp\) 序列里有多少个 \(0\),所以需要再开个数组 \(g[i][j]\) 表示没有元素为 \(0\) 的方案数。如何让元素不为 \(0\) 呢?把每个元素 \(+1\) 即可。于是有:

\[\begin{split}
dp[i][j]&=dp[i][j-i]+\sum_{k=1}^{m}dp[i-k][j]\\
g[i][j]&=dp[i][j-1]
\end{split}
\]

如果直接 for 硬套的话,复杂度为 \(O(N^2M)\)。会超。

70 TLE 代码:

#include<bits/stdc++.h>
using namespace std;
#define int long long
constexpr int N = 5010, M = 998244353;
int n, m, dp[N][N], g[N][N];
signed main(){
ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
cin>>n>>m;
for(int i=1; i<=m; ++i) dp[i][0] = 1ll;
for(int i=1; i<=n; ++i){
for(int j=1; j<=n; ++j){
if(j >= i){
g[i][j] = (g[i][j] + dp[i][j-i]) % M;
dp[i][j] = (dp[i][j] + dp[i][j-i]) % M;
}
for(int k=1; k<=m && i-k>=0; ++k){
dp[i][j] = (dp[i][j] + g[i-k][j]) % M;
}
}
}
for(int i=1; i<=n; ++i) cout<<g[i][n]<<'\n';
return 0;
}

对于那个最内层循环,可以使用前缀和优化,使复杂度分摊到每一次循环 \(O(1)\),于是总复杂度 \(O(N^2)\),对了,本题不能开 long long,会爆空间。

AC code

#include<bits/stdc++.h>
using namespace std;
constexpr int N = 5001, M = 998244353;
int n, m, dp[N][N], g[N][N], sum[N][N];
int main(){
ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
cin>>n>>m;
for(int i=1; i<=m; ++i) dp[i][0] = 1ll;
for(int i=1; i<=n; ++i){
for(int j=1; j<=n; ++j){
if(j >= i){
g[i][j] = (g[i][j] + dp[i][j-i]) % M;
dp[i][j] = (dp[i][j] + dp[i][j-i]) % M;
}
sum[i][j] = (sum[i-1][j] + g[i][j]) % M;
dp[i][j] = (__int128)(dp[i][j] + sum[i-1][j] - sum[max(i-m-1, 0)][j]) % M;
}
}
for(int i=1; i<=n; ++i) cout<<g[i][n]<<'\n';
return 0;
}

小结:

  • 不要过于思维定式。如果发现一个方案不可行或很难行要跳出来另找其他方案。
  • 对于求转移方程,可以考虑控制变量,即只对某一变量产生贡献,最后再将贡献合并。很好的思路,只是比较难想。
  • define int long long 不要再用了。
  • 计算空间复杂度。

[题解] [ABC221H] Count Multiset - DP的更多相关文章

  1. 【题解】POJ1934 Trip (DP+记录方案)

    [题解]POJ1934 Trip (DP+记录方案) 题意: 传送门 刚开始我是这么设状态的(谁叫我DP没学好) \(dp(i,j)\)表示钦定选择\(i\)和\(j\)的LCS,然而你会发现这样钦定 ...

  2. 【题解】剪纸条(dp)

    [题解]剪纸条(dp) HRBUST - 1828 网上搜不到题解?那我就来写一篇吧哈哈哈 最优化问题先考虑\(dp\),设\(dp(i)\)表示将前\(i\)个字符(包括\(i\))分割成不相交的回 ...

  3. 【题解】地精部落(DP)

    [题解]地精部落(DP) 设\(f_i\)表示强制第一个是谷的合法方案数 转移枚举一个排列的最大值在哪里,就把序列分成了互不相干的两个部分,把其中\(i-1\choose j-1\)的数字分配给前面部 ...

  4. LeetCode题解38.Count and Say

    38. Count and Say The count-and-say sequence is the sequence of integers beginning as follows: 1, 11 ...

  5. bzoj 2669 题解(状压dp+搜索+容斥原理)

    这题太难了...看了30篇题解才整明白到底咋回事... 核心思想:状压dp+搜索+容斥 首先我们分析一下,对于一个4*7的棋盘,低点的个数至多只有8个(可以数一数) 这样的话,我们可以进行一个状压,把 ...

  6. noi省选 [九省联考2018]一双木棋题解(状压dp)

    比浙江简单多了........ 题目转送:https://www.luogu.org/problemnew/show/P4363 分析: 我们注意到n和m都很小,考虑一下状压dp. 显然,棋子摆成的形 ...

  7. 『count 区间dp』

    count Description 既然是萌萌哒 visit_world 的比赛,那必然会有一道计数题啦! 考虑一个N个节点的二叉树,它的节点被标上了1-N的编号. 并且,编号为i的节点在二叉树的前序 ...

  8. 理想乡题解 (线段树优化dp)

    题面 思路概述 首先,不难想到本题可以用动态规划来解,这里就省略是如何想到动态规划的了. 转移方程 f[i]=min(f[j]+1)(max(i-m,0)<=j<i 且j符合士兵限定) 注 ...

  9. CF 1400F x-prime Substrings 题解【AC自动机+DP】

    CF 1400F.x-prime Substrings 题意: 给定一个由\('1'\)到\('9'\)组成的字符串\(s\)和一个数\(x\),定义一个串为\(x-prime\)串,当且仅当这个串上 ...

  10. BZOJ 1087 题解【状压DP】

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3112  Solved: 1816[Submit][ ...

随机推荐

  1. 免费的Java主流jdk发行版本有哪些?

    Java的特点是百花齐放,不像c#或者go只有一家主导.oracle jdk收费了,没关系,不是只有它可用.java还有很多免费的主流的jdk发行版本,记录下来备忘. OpenJDK - 官方网站 - ...

  2. arm linux 移植 SQLite 3

    背景 SQLite 是 一个 常用于 嵌入式平台的 轻量级数据库. host平台 :Ubuntu 16.04 arm平台 : S5P6818 SQLite :3.31.1 arm-gcc :4.8.1 ...

  3. React Lazy 和 Suspense

    在React应用中,有些组件可能不经常用到,比如法律条款的弹窗,我们几乎不看,这些组件也就没有必要首次加载,可以在点击它们的时候再加载,这就需要动态引入组件,需要组件的时候,才引入组件,加载它们,进行 ...

  4. Spring注解之依赖注入@Autowired和@Resource

    Spring常见的DI方式 字段注入(Field Injection) 在字段上使用@Autowired/Resource注解 字段注入是日常开发中使用最多的一种注入方式,它的实现代码如下: @Aut ...

  5. MySQL 并发控制(锁得使用)

    导读 并发问题:同一时刻进行读写,并发问题回引发数据不一致问题. 解决并发问题:MySQL采用了锁定机制去解决并发问题 锁的分类 MySQL使用两种锁机制去解决问题:共享锁和排他锁,也叫读锁或者写锁. ...

  6. 前端:如何让background背景图片进行CSS自适应

    在设置login背景时,找到了一张这样的图片: 但是设置成login背景时,如果没有做一些css适应设置,图片就变样了,变成了这样: 严重变形了,这就造成了一种理想与现实的差距. 若想解决这个自适应问 ...

  7. 算法金 | 最难的来了:超参数网格搜索、贝叶斯优化、遗传算法、模型特异化、Hyperopt、Optuna、多目标优化、异步并行优化

    ​ 大侠幸会,在下全网同名「算法金」 0 基础转 AI 上岸,多个算法赛 Top 「日更万日,让更多人享受智能乐趣」 今日 215/10000 为模型找到最好的超参数是机器学习实践中最困难的部分之一 ...

  8. [oeasy]python0037_电传打字机_打印头_print_head_carriage_词源

    换行回车 回忆上次内容 上次我们 diy了 自己的小动物 还可以 让小动物 变色.报时 还可以 说些话 这很亚文化 很酷炫的亚文化 不是吗? 回忆一下 最开始 研究报时 的 时候 回到 本行行头 的 ...

  9. 题解:B3646 数列前缀和 3

    分析 板子题,线段树维护矩阵区间积,除了难写没什么思维难度. 所以直接放代码吧. Code #include<bits/stdc++.h> #define int long long us ...

  10. 栈—顺序栈(C实现)

    // Code file created by C Code Develop // 顺序栈 #include "ccd.h" #include "stdio.h" ...