@description@

有 \(n\) 个在范围 \([1, D]\) 内的整数均匀随机变量。

求至少能选出 \(m\) 个瓶子,使得存在一种方案,选择一些变量,并把选出来的每一个变量放到一个瓶子中,满足每个瓶子都恰好装两个值相同的变量的概率。

请输出概率乘上 \(D^n\) 后对 \(998244353\) 取模的值。

原题传送门。

@solution@

记 \(l = \min\{n - 2m, D\}\)。不难想到枚举出现次数为奇数的变量个数 \(i\),列式子:

\[n!\times[x^n]\sum_{i=0}^{l}{D\choose i}(\frac{e^x - e^{-x}}{2})^i(\frac{e^x + e^{-x}}{2})^{D-i}
\]

然后二项式展开 + 整理化简得到:

\[\frac{D!}{2^D}\times\sum_{i=0}^{D}\sum_{p=0}^{i}\sum_{q=0}^{D-i} \frac{(-1)^{i-p}(2p+2q-D)^n}{p!q!(i-p)!(D-i-q)!}
\]

然后我就到此为止了。不过这个思路是可以继续往下推导的,可以参考这篇博客。出题人也有一个关于这种推导方法的解释(还没仔细看,改天补坑.jpg)。

有一个较简单的推导方法。上式中出现了 3 个 \(\sum\),远远超过了一个中老年菜鸡选手的承受范围。

我们不妨设法去掉 1 个,然后再用常规卷积方法优化。

不妨钦定选 i 个奇数变量,其他的任意选,得到表达式:

\[\begin{aligned}
g_i &= n!\times{D \choose i}\times [x^n](\frac{e^x - e^{-x}}{2})^ie^{(D-i)x} \\
&= n!\times{D \choose i}\times [x^n]\sum_{j=0}^i{i\choose j}e^{(2j-2i+D)x}
\end{aligned}
\]

可以发现我们少了一次二项式展开。把 \({i\choose j}\) 拆开就可以直接卷积了。

设 \(f_i\) 表示恰好 i 个奇数变量的方案数,那么有 \(g_i = \sum_{j=i}^{D}{j\choose i}f_j\),二项式反演一下:

\[f_n = \sum_{i=n}^{D}(-1)^{i-n}{i\choose n}g_i
\]

这玩意也可以卷积。然后就做完了。

@accepted code@

#include <cstdio>
#include <algorithm>
using namespace std; const int MOD = 998244353;
const int MAXN = 400000; inline int add(int x, int y) {x += y; return x >= MOD ? x - MOD : x;}
inline int sub(int x, int y) {x -= y; return x < 0 ? x + MOD : x;}
inline int mul(int x, int y) {return 1LL * x * y % MOD;} int pow_mod(int b, int p) {
int ret = 1;
for(int i=p;i;i>>=1,b=mul(b,b))
if( i & 1 ) ret = mul(ret, b);
return ret;
} int w[20], iw[20], fct[MAXN + 5], ifct[MAXN + 5], inv[MAXN + 5];
void init() {
for(int i=0;i<20;i++) {
w[i] = pow_mod(3, (MOD - 1) / (1 << i));
iw[i] = pow_mod(w[i], MOD - 2);
}
fct[0] = 1; for(int i=1;i<=MAXN;i++) fct[i] = mul(fct[i - 1], i);
ifct[MAXN] = pow_mod(fct[MAXN], MOD - 2);
for(int i=MAXN-1;i>=0;i--) ifct[i] = mul(ifct[i + 1], i + 1);
for(int i=1;i<=MAXN;i++) inv[i] = mul(fct[i - 1], ifct[i]);
}
void ntt(int *A, int n, int type) {
for(int i=0,j=0;i<n;i++) {
if( i < j ) swap(A[i], A[j]);
for(int l=(n>>1);(j^=l)<l;l>>=1);
}
for(int i=1,s=2,t=1;s<=n;i++,s<<=1,t<<=1) {
int u = (type == 1 ? w[i] : iw[i]);
for(int j=0;j<n;j+=s) {
for(int k=0,p=1;k<t;k++,p=mul(p,u)) {
int x = A[j + k], y = mul(p, A[j + k + t]);
A[j + k] = add(x, y), A[j + k + t] = sub(x, y);
}
}
}
if( type == -1 ) {
for(int i=0;i<n;i++)
A[i] = mul(A[i], inv[n]);
}
}
int length(int n) {
int len; for(len = 1; len <= n; len <<= 1);
return len;
}
int comb(int n, int m) {
return mul(fct[n], mul(ifct[m], ifct[n - m]));
}
int D, n, m;
int f[MAXN + 5], g[MAXN + 5], a[MAXN + 5], b[MAXN + 5];
void get() {
for(int i=0;i<=D;i++) a[i] = mul((i & 1 ? MOD - 1 : 1), mul(pow_mod(sub(D, 2*i), n), ifct[i]));
for(int i=0;i<=D;i++) b[i] = ifct[i];
int len = length(2*D); ntt(a, len, 1), ntt(b, len, 1);
for(int i=0;i<len;i++) g[i] = mul(a[i], b[i]);
ntt(g, len, -1);
for(int i=1;i<=D;i++) g[i] = mul(g[i], mul(comb(D, i), mul(fct[i], pow_mod(2, MOD - 1 - i))));
for(int i=D+1;i<len;i++) g[i] = 0; for(int i=0;i<len;i++) a[i] = b[i] = 0;
for(int i=0;i<=D;i++) a[D - i] = mul(g[i], fct[i]);
for(int i=0;i<=D;i++) b[i] = (i & 1 ? sub(0, ifct[i]) : ifct[i]);
ntt(a, len, 1), ntt(b, len, 1);
for(int i=0;i<len;i++) f[i] = mul(a[i], b[i]);
ntt(f, len, -1), reverse(f, f + D + 1);
for(int i=0;i<=D;i++) f[i] = mul(f[i], ifct[i]);
} int main() {
init(), scanf("%d%d%d", &D, &n, &m), get();
int l = min(D, n - 2*m), ans = 0;
for(int i=0;i<=l;i++) ans = add(ans, f[i]);
printf("%d\n", ans);
}

@details@

感觉第二种做法比第一种做法更自然一些?(虽然我觉得一般都会先想到第一种做法)

看了一下某谷的题解,发现好像还有线性做法?

我感觉我的组合数学现在可能只会卷积模板题了。

@loj - 3120@ 「CTS2019 | CTSC2019」珍珠的更多相关文章

  1. 【LOJ】#3120. 「CTS2019 | CTSC2019」珍珠

    LOJ3120 52pts \(N - D >= 2M\)或者\(M = 0\)那么就是\(D^{N}\) 只和数字的奇偶性有关,如果有k个奇数,那么必须满足\(N - k >= 2M\) ...

  2. Loj #3124. 「CTS2019 | CTSC2019」氪金手游

    Loj #3124. 「CTS2019 | CTSC2019」氪金手游 题目描述 小刘同学是一个喜欢氪金手游的男孩子. 他最近迷上了一个新游戏,游戏的内容就是不断地抽卡.现在已知: - 卡池里总共有 ...

  3. LOJ 3120: 洛谷 P5401: 「CTS2019 | CTSC2019」珍珠

    题目传送门:LOJ #3120. 题意简述: 称一个长度为 \(n\),元素取值为 \([1,D]\) 的整数序列是合法的,当且仅当其中能够选出至少 \(m\) 对相同元素(不能重复选出元素). 问合 ...

  4. LOJ 3124 「CTS2019 | CTSC2019」氪金手游——概率+树形DP

    题目:https://loj.ac/problem/3124 看了题解:https://www.cnblogs.com/Itst/p/10883880.html 先考虑外向树. 考虑分母是 \( \s ...

  5. loj3120 「CTS2019 | CTSC2019」珍珠

    link .... 感觉自己太颓废了....还是来更题解吧...[话说写博客会不会涨 rp 啊 qaq ? 题意: 有 n 个物品,每个都有一个 [1,D] 中随机的颜色,相同颜色的两个物品可以配对. ...

  6. LOJ #3119「CTS2019 | CTSC2019」随机立方体 (容斥)

    博客链接 里面有个下降幂应该是上升幂 还有个bk的式子省略了k^3 CODE 蛮短的 #include <bits/stdc++.h> using namespace std; const ...

  7. LOJ #3119. 「CTS2019 | CTSC2019」随机立方体 组合计数+二项式反演

    好神的一道计数题呀. code: #include <cstdio> #include <algorithm> #include <cstring> #define ...

  8. 「CTS2019 | CTSC2019」氪金手游 解题报告

    「CTS2019 | CTSC2019」氪金手游 降 智 好 题 ... 考场上签到失败了,没想容斥就只打了20分暴力... 考虑一个事情,你抽中一个度为0的点,相当于把这个点删掉了(当然你也只能抽中 ...

  9. 「CTS2019 | CTSC2019」随机立方体 解题报告

    「CTS2019 | CTSC2019」随机立方体 据说这是签到题,但是我计数学的实在有点差,这里认真说一说. 我们先考虑一些事实 如果我们在位置\((x_0,y_0,z_0)\)钦定了一个极大数\( ...

随机推荐

  1. 转义URL 含有中文和特殊符号

    方法1: //这个方法被废弃了 NSString *urlString = @"https://www.cnblogs.com/huaida/#/程序员"; NSString* e ...

  2. Educational Codeforces Round 56 (Rated for Div. 2) F. Vasya and Array

    题意:长度为n的数组,数组中的每个元素的取值在1-k的范围内或者是-1,-1代表这个元素要自己选择一个1-k的数字去填写,然后要求填完的数组中不能出现连续长度大于len的情况,询问填空的方案数. 题解 ...

  3. CF861D

    题目链接:http://codeforces.com/contest/861/problem/D 解题思路: 优雅的暴力. 对于输入的每一个号码,从短到长找出它的所有子串,用 vector 保存每个号 ...

  4. Golang源码学习:监控线程

    监控线程是在runtime.main执行的时候在系统栈中创建的,监控线程与普通的工作线程区别在于,监控线程不需要绑定p来运行. 监控线程的创建与启动 简单的调用图 先给出个简单的调用图,好心里有数,逐 ...

  5. JavaScript实现树深度优先和广度优先遍历搜索

    1.前置条件 我们提前构建一棵树,类型为 Tree ,其节点类型为 Note.这里我们不进行过多的实现,简单描述下 Note 的结构: class Node{ constructor(data){ t ...

  6. [JavaWeb基础] 009.Struts2 上传文件

    在web开发中,我们经常遇到要把文件上传下载的功能,这篇文章旨在指导大家完成文件上传功能 1.首先我们需要一个上传文件的页面. <!--在进行文件上传时,表单提交方式一定要是post的方式, 因 ...

  7. Linux(二):VMware虚拟机中Ubuntu安装详细过程

    Linux(二):VMware虚拟机中Ubuntu安装详细过程 目录 1 准备 2 安装 2.1 虚拟机的建立 2.2 虚拟机安装Ubuntu系统 2.3 虚拟机设置 3 完成 1 准备 1.操作系统 ...

  8. 2020 网鼎杯wp

    2020 网鼎杯WP 又是划水的一天,就只做出来4题,欸,还是太菜,这里就记录一下做出的几题的解题记录 AreUSerialz 知识点:反序列化 打开链接直接给出源码 <?php include ...

  9. 09 . Python3之常用模块

    模块的定义与分类 模块是什么? 一个函数封装一个功能,你使用的软件可能就是由n多个函数组成的(先备考虑面向对象).比如抖音这个软件,不可能将所有程序都写入一个文件,所以咱们应该将文件划分,这样其组织结 ...

  10. PHP AES加解密(兼容php5,php7)

    最近在工作中负责对接API,对方要求对业务数据进行AES 算法(256,ECB,补码方式:PKCS5Padding)加密. 加密算法要求如下: 算法AES/ECB/PKCS5Padding 密钥长度2 ...