Solution -「Gym 102798E」So Many Possibilities...
\(\mathcal{Description}\)
Link.
给定非负整数序列 \(\{a_n\}\) 和 \(m\),每次随机在 \(\{a\}\) 中取一个非零的 \(a_i\)(保证存在),令其 \(-1\),重复 \(m\) 次,求最终 \(\{a\}\) 中 \(0\) 的期望个数。
\(n\le15\),\(m\le100\)。
\(\mathcal{Solution}\)
既然有 DP of DP,我觉得这种题就叫 DP and DP。(
看到 \(n\le15\),想要状压,但无论如何都不能完整记录 \(\{a\}\) 的状态,也就不可能及时知道某个 \(a\) 会否变成 \(0\),怎么办呢?
既然不知道,我们索性不去动它们——设前 \(i\) 次操作后,\(\{a\}\) 中为 \(0\) 的下标集合为 \(S\), 我们只落实对 \(S\) 中元素造成影响的操作,共 \(\sum_{i\in S}a_i\) 次;而对于剩下 \(\left(i-\sum_{i\in S}a_i\right)\) 次,只记录为“它们会影响某个不在 \(T\) 中元素”,其中 \(T\) 是进行这个操作时的 \(0\) 值集合;“某个”是具体的,但尚未确定的一个,所以提供概率 \(\frac{1}{n-|T|}\)。另一方面,只有在扩充 \(S\) 时来确定若干“某个”,此时就只需要组合选取,而不需要乘概率了。那么摆出式子,令 \(f(i,S)\) 表示 \(i\) 次操作后,\(0\) 值集合为 \(S\) 的概率(概率计算方式如前文),则:
\]
思考此时得到的,应该用于计算答案的 \(f(m,S)\) 意味什么:落在 \(S\) 内的操作,它们是正常的,而其余的 \(\left(m-\sum_{i\in S}a_i\right)\) 次,它们一定落在 \(U\setminus S\)(\(U\) 为全集)中,遗漏的仅仅是方案数!所以在 \(f\) 的目标状态为基础,再来一个 DP:令 \(g(i,S)\) 表示有 \(i\) 次操作落在 \(S\),且 \(S\) 中不存在 \(0\) 值的方案数,简单地,任取一个 \(x\in S\),得到转移:
\]
最终答案则为:
\]
复杂度为 \(\mathcal O(m(n+m)2^n)\)。
\(\mathcal{Code}\)
/*~Rainybunny~*/
#include <cstdio>
#define rep( i, l, r ) for ( int i = l, rep##i = r; i <= rep##i; ++i )
#define per( i, r, l ) for ( int i = r, per##i = l; i >= per##i; --i )
inline int imin( const int a, const int b ) { return a < b ? a : b; }
typedef long double LD;
#define double LD
const int MAXN = 15, MAXM = 100;
int n, m, a[MAXN + 5], sum[1 << MAXN], bitc[1 << MAXN], bitw[1 << MAXN];
double comb[MAXM + 5][MAXM + 5], f[2][1 << MAXN], g[MAXM + 5][1 << MAXN];
inline void init() {
comb[0][0] = 1.;
rep ( i, 1, m ) {
comb[i][0] = 1.;
rep ( j, 1, i ) comb[i][j] = comb[i - 1][j] + comb[i - 1][j - 1];
}
}
inline void getF() {
f[0][0] = 1.;
rep ( i, 1, ( 1 << n ) - 1 ) bitc[i] = bitc[i ^ ( i & -i )] + 1;
for ( int i = 1, sta = 1; i <= m; ++i, sta ^= 1 ) {
rep ( S, 0, ( 1 << n ) - 1 ) {
f[sta][S] = 0.;
rep ( j, 0, n - 1 ) if ( S >> j & 1 ) {
f[sta][S] += i < sum[S ^ 1 << j] + 1 ? 0 :
comb[i - sum[S ^ 1 << j] - 1][a[j] - 1] * f[!sta][S ^ 1 <<j];
}
f[sta][S] /= n - bitc[S] + 1;
if ( n > bitc[S] ) f[sta][S] += f[!sta][S] / ( n - bitc[S] );
// fprintf( stderr, "f(%d,%d)=%f\n", i, S, f[sta][S] );
}
}
}
inline void getG() {
rep ( i, 2, ( 1 << n ) - 1 ) bitw[i] = bitw[i >> 1] + 1;
rep ( S, 0, ( 1 << n ) - 1 ) g[0][S] = 1.;
rep ( i, 1, m ) {
rep ( S, 1, ( 1 << n ) - 1 ) {
int v = bitw[S & -S]; double &cur = g[i][S];
rep ( j, 0, imin( i, a[v] - 1 ) ) {
cur += g[i - j][S ^ 1 << v] * comb[i][j];
}
// fprintf( stderr, "g(%d,%d)=%.0f\n", i, S, g[i][S] );
}
}
}
int main() {
scanf( "%d %d", &n, &m ), init();
rep ( i, 0, n - 1 ) scanf( "%d", &a[i] ), sum[1 << i] = a[i];
rep ( S, 1, ( 1 << n ) - 1 ) sum[S] = sum[S & -S] + sum[S ^ ( S & -S )];
getF(), getG();
double ans = 0.;
rep ( S, 0, ( 1 << n ) - 1 ) if ( m >= sum[S] ) {
ans += f[m & 1][S] * g[m - sum[S]][repS ^ S] * bitc[S];
}
printf( "%.12Lf\n", ans );
return 0;
}
Solution -「Gym 102798E」So Many Possibilities...的更多相关文章
- Solution -「Gym 102979E」Expected Distance
\(\mathcal{Description}\) Link. 用给定的 \(\{a_{n-1}\},\{c_n\}\) 生成一棵含有 \(n\) 个点的树,其中 \(u\) 连向 \([1, ...
- Solution -「Gym 102979L」 Lights On The Road
\(\mathcal{Description}\) Link. 给定序列 \(\{w_n\}\),选择 \(i\) 位置的代价为 \(w_i\),要求每个位置要不被选择,要不左右两个位置至少被 ...
- Solution -「Gym 102956F」Find the XOR
\(\mathcal{Description}\) Link. 给定 \(n\) 个点 \(m\) 条边的连通无向图 \(G\),边有边权.其中 \(u,v\) 的距离 \(d(u,v)\) ...
- Solution -「Gym 102956B」Beautiful Sequence Unraveling
\(\mathcal{Description}\) Link. 求长度为 \(n\),值域为 \([1,m]\) 的整数序列 \(\lang a_n\rang\) 的个数,满足 \(\not\ ...
- Solution -「Gym 102956F」Border Similarity Undertaking
\(\mathcal{Description}\) Link. 给定一张 \(n\times m\) 的表格,每个格子上写有一个小写字母.求其中长宽至少为 \(2\),且边界格子上字母相同的矩 ...
- Solution -「Gym 102956A」Belarusian State University
\(\mathcal{Description}\) Link. 给定两个不超过 \(2^n-1\) 次的多项式 \(A,B\),对于第 \(i\in[0,n)\) 个二进制位,定义任意一个二元 ...
- Solution -「Gym 102798I」Sean the Cuber
\(\mathcal{Description}\) Link. 给定两个可还原的二阶魔方,求从其中一个状态拧到另一个状态的最小步数. 数据组数 \(T\le2.5\times10^5\). ...
- Solution -「Gym 102798K」Tree Tweaking
\(\mathcal{Description}\) Link. 给定排列 \(\{p_n\}\),求任意重排 \(p_{l..r}\) 的元素后,将 \(\{p_n\}\) 依次插入二叉搜索树 ...
- Solution -「Gym 102759I」Query On A Tree 17
\(\mathcal{Description}\) Link. 给定一棵含 \(n\) 个结点的树,结点 \(1\) 为根,点 \(u\) 初始有点权 \(a_u=0\),维护 \(q\) 次 ...
随机推荐
- git branch --set-upstream-to 本地关联远程分支
最近使用git pull的时候多次碰见下面的情况: There is no tracking information for the current branch. Please specify wh ...
- Java安全之Spring内存马
Java安全之Spring内存马 基础知识 Bean bean 是 Spring 框架的一个核心概念,它是构成应用程序的主干,并且是由 Spring IoC 容器负责实例化.配置.组装和管理的对象. ...
- 使用 Jenkins + Ansible 实现 Spring Boot 自动化部署101
本文要点:设计一条 Spring Boot 最基本的流水线:包括构建.制品上传.部署.使用 Docker 容器运行构建逻辑.自动化整个实验环境:包括 Jenkins 的配置,Jenkins agent ...
- 不会一致性hash算法,劝你简历别写搞过负载均衡
大家好,我是小富~ 个人公众号:程序员内点事,欢迎学习交流 这两天看到技术群里,有小伙伴在讨论一致性hash算法的问题,正愁没啥写的题目就来了,那就简单介绍下它的原理.下边我们以分布式缓存中经典场景举 ...
- hdfs文件导入到hive(带资源)
前言 hive是基于Hadoop的一个数据仓库工具,用来进行数据提取.转化.加载,这是一种可以存储.查询和分析存储在Hadoop中的大规模数据的机制.hive数据仓库工具能将结构化的数据文件映射为一张 ...
- C# 实现Parallel.For
static class MyParallel { //4.0及以上用Task, Task的背后的实现也是使用了线程池线程 //static List<Task> tasks = new ...
- iptables匹配条件总结1
源地址 -s选项除了指定单个IP,还可以一次指定多个,用"逗号"隔开即可 [root@web-1 ~]# iptables -I INPUT -s 172.16.0.116,172 ...
- linux挂载windows nfs
1.win下创建nfs文件夹并共享 2.登陆linux,执行 yum 3.创建挂载点 4.挂载win nfs 5./etc/fstab添加永久挂载 6.查看挂载磁盘,此时windows盘已落在linu ...
- python33day
内容回顾 概念 同步异步阻塞和非阻塞 同步阻塞:调用一个函数需要等待这个函数的执行结果,并且在执行这个函数的过程中CPU不工作 inp=input('>>>') 同步非阻塞:调用一个 ...
- Java多线程专题3: Thread和ThreadLocal
合集目录 Java多线程专题3: Thread和ThreadLocal 进程, 线程, 协程的区别 进程 Process 进程提供了执行一个程序所需要的所有资源, 一个进程的资源包括虚拟的地址空间, ...