CodeForces 261B Maxim and Restaurant 解法汇总
题意:给定n个数a1…an(n<=50,ai<=50),随机打乱后,记Si=a1+a2+a3…+ai,问满足Si<=p的i的最大值的期望.(p<=50)
这道题在网上有一些不同的做法,O(n^3)或O(n^4)都可以通过,这里整合一下,标上出处,其实我只写了自己YY的那一种,叫我搬运工
1.期望的线性性,讨论每个数对i的贡献.O(n^4)
自己YY的,不知道以前有没有人也写过这种方法.
如果ai满足Si<=p,那么ai就对答案有1 的贡献,因此我们算出每个数ai满足Si<=p的概率(其实也是对答案贡献的期望),加起来就是答案.
现在对于每个数考虑满足条件的概率,我们不妨考虑将这个数先放进序列里,再随机往序列里加数.如果现在这个数的前面有i个数,后面有j个数,那么再随机加一个数位于这个数前面的概率是(i+1)/(i+1+j+1),(j+1)/(i+1+j+1),因为这个数的前面有i+1个空,后面有j+1个空,而且新加的数插到每个空的概率是相同的.
那么我们定义F[i][j][s]表示这个数前面有i个数,后面有j个数,前面的数的和为s的情况在构造整个序列的中间过程中出现的概率,则F[i][j][s]=F[i][j-1][s]*p(最后一个数插在后面)+F[i-1][j][s-a[last]]*p(最后一个数插在前面),其中p(最后一个数插在后面)=(j)/(i+j+1),
p(最后一个数插在前面)=(i+1)/(i+j+1).边界F[0][0][0]=1,也就是一开始我们的序列里只有一个数.
于是我们做n次O(n^3)的DP即可.
看上去我们每次DP生成这个随机序列的方式不同,但这些方式其实是等价的,因为每个数都是随机插入.
写的时候可以通过恰当的循环顺序把DP数组只开两维,不过并没有卡内存.
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
double f[][];//f[i][j]:前面有j个人,长度之和为i
int a[];
int lim,n;
double dp(int x){
memset(f,,sizeof(f));
f[][]=;
int flag=;
for(int i=;i<=n;++i){
if(i==x){
flag=;continue;
}
for(int j=;j>=;--j){
for(int k=;k>=;--k){
f[min(j+a[i],)][k+]+=f[j][k]*(k+)/(i-flag+);
f[j][k]=f[j][k]*(i-flag-k)/(i-flag+);
}
}//printf("%.3f\n",f[0][0]);
}
double ans=;
for(int i=;i+a[x]<=lim;++i)
for(int j=;j<=n;++j)//printf("%d %d %d %.3f\n",x,i,j,f[i][j])
ans+=f[i][j];
// printf("%.3f\n",ans);
return ans;
}
int main(){
scanf("%d",&n);
for(int i=;i<=n;++i){
scanf("%d",a+i);
}
scanf("%d",&lim);
double ans=;
for(int i=;i<=n;++i)ans+=dp(i);
printf("%.5f\n",ans);
return ;
}
2.期望的定义,O(n^4)
传送门:http://blog.csdn.net/hmzhe/article/details/51960298
我们知道所有可能的排列一共有n!种,50!虽然是一个很大的数字,但double可以提供足够的精度(有一定的精度损失,但最终结果损失的精度在题目要求的1e-4之外).因此我们可以算出所有排列中的答案之和再除以n!就是答案.
显然我们不能用阶乘的时间复杂度真正地枚举所有答案.因此我们需要将某些具有相同特征的排列一并计算.注意到,如果一个方案在Si<=p时只算进去前m个数,那么后(n-m)个数如何排列我们是不关心的.也就是说我们可以得到一共(n-m)!个方案.
那么我们只需要考虑算了0个数的方案有几个,算了1个数的方案有几个,算了2个数的方案有几个….这里的方案数也可能很大,需要double(最终结果会炸掉不影响答案正确性的1e-4之后精度,中间炸的精度可能大一点但除以n!后就变到1e-4之后了).
此时我们需要确保我们所计数的”算了一个数的方案”不会在加一个数之后变成”算了两个数的方案”,所以我们需要确保这个方案所选择的前缀在末尾加了一个数之后和大于p,也就是我们需要定义F[i][j][s][ed]为在前i个数中选取j个和为s的数,且在这些数的最后加上一个数ed后超过限制的方案数.这个ed似乎不太好处理,所以我们应当在最外层枚举它.最后只有s<p且s+a[ed]>p的状态合法.x后面的数字自由排列,所以方案数乘上(n-j-1)!如果在计算选取方案的时候没有考虑顺序,最后需要对前面j个数也进行排列,方案数乘上j!
3.期望的线性性,O(n^3)
传送门:http://blog.sina.com.cn/s/blog_140e100580102wj4e.html
做法1.中我们考虑了原序列中每个数对期望的贡献.实际上我们可以转而考虑最终序列中每个位置上的数对期望的贡献,那么不需要考虑每种方案的截止位置,只需要考虑无序选出i个数和为j(j<p)的方案数x即可得出第i个位置对最终答案的贡献x*(i!)*(n-i)!/n!
4.期望的线性性,O(n^3)
codeforces别人的一份AC代码:http://codeforces.com/contest/261/submission/2917070
可以发现我们在做法3中算的其实是每个位置对答案贡献的期望.直接定义F[i][j][k]为在原序列的前i个数中选取了最终序列的前j个数,且和为k的情况在所有情况中所占的概率.写法和3相似.
CodeForces 261B Maxim and Restaurant 解法汇总的更多相关文章
- codeforces 261B Maxim and Restaurant(概率DP)
B. Maxim and Restaurant time limit per test 2 seconds memory limit per test 256 megabytes input stan ...
- CodeForces - 261B Maxim and Restaurant
http://codeforces.com/problemset/problem/261/B 题目大意:给定n个数a1-an(n<=50,ai<=50),随机打乱后,记Si=a1+a2+a ...
- cf 261B.Maxim and Restaurant
什么什么期望的,不会! (题解http://blog.sina.com.cn/s/blog_140e100580102wj4e.html(看不懂)) #include<bits/stdc++.h ...
- 【CodeForces 261B】Maxim and Restaurant(DP,期望)
题目链接 第一种解法是$O(n^3*p)$的:f[i][j][k]表示前i个人进j个人长度为k有几种方案(排列固定为123..n时).$f[i][j][k]=f[i-1][j][k]+f[i-1][j ...
- Codeforces Round #160 (Div. 2) D. Maxim and Restaurant(DP)
题目链接 想了挺久,枚举每一件物品,当做关键物品,假设再加这一件物品,就>=c了,把剩下的物品背一下包,dp[i][j]表示i个物品可以组成重量j的个数. 这样就可以知道前面放i件,后边肯定放n ...
- 整数划分问题-解法汇总(暂有DP-递归)
整数划分问题是一个锻炼组合数学,递归以及动态规划很好的例子,虽然问题看似简单,但是其中玄机万千,有人转化成为背包问题,有人用生成函数解,有人以此作为企业面试题目,可见这种问题的认可度还是很高的. 整数 ...
- Codeforces Round #423 A Restaurant Tables(模拟)
A. Restaurant Tables time limit per test 1 second memory limit per test 256 megabytes input standard ...
- Codeforces 854B Maxim Buys an Apartment:贪心
题目链接:http://codeforces.com/contest/854/problem/B 题意: 有n栋房子从1到n排成一排,有k栋房子已经被售出. 现在你要买一栋“好房子”. 一栋房子是“好 ...
- Codeforces F. Maxim and Array(构造贪心)
题目描述: Maxim and Array time limit per test 2 seconds memory limit per test 256 megabytes input standa ...
随机推荐
- datatables中的Options总结(2)
datatables中的Options总结(2) 五.datatable,列 columnDefs.targets 分配一个或多个列的列定义. columnDefs 设置列定义初始化属性. colum ...
- swing with transformjs
Antecedent Facebook made a HTML5 game long time ago. The opening animation is a piece of software th ...
- 前端导出Excel兼容写法
今天整理出在Web前端导出Excel的写法,写了一个工具类,对各个浏览器进行了兼容. 首先,导出的数据来源可能有两种: 1. 页面的HTML内容(一般是table) 2. 纯数据 PS:不同的数据源, ...
- Hadoop学习日志- install hadoop
资料来源 : http://www.tutorialspoint.com/hadoop/hadoop_enviornment_setup.htm Hadoop 安装 创建新用户 $ su passwo ...
- LDAP注入与防御解析
[目录] 0x1 LDAP介绍 0x2 LDAP注入攻击及防御 0x3 参考资料 0x1 LDAP介绍 1 LDAP出现的背景 LDAP(Lightweight Directory Access Pr ...
- Linux系统安装MySql步骤及截屏
➠更多技术干货请戳:听云博客 如下是我工作中的记录,介绍的是linux系统下使用官方编译好的二进制文件进行安装MySql的安装过程和安装截屏,这种安装方式速度快,安装步骤简单! 需要的朋友可以按照如下 ...
- Oracle工具类-生成数据库现有Job的创建脚本
生成Oracle数据库现有Job的创建脚本 -- 生成现有Job的创建脚本 create or replace procedure proc_generate_job_create_sql is be ...
- bug描述技巧
进入测试行业已经两年了,我从未认真的考虑过提交一个bug需要注意哪些问题,只是主观的认为我只需要描述清楚就OK了,但是我在工作中发现有个别的开发经常跑来告诉我"这个bug你是不是描述错了&q ...
- Java使用MyEclipse构建webService简单案例
什么是WebServices? 它是一种构建应用程序的普遍模型,可以在任何支持网络通信的操作系统中实施运行;它是一种新的web应用程序分支,是自包含.自描述.模块化的应用,可以发布.定位.通过web ...
- 【转】BAT 批处理脚本 教程
第一章 批处理基础第一节 常用批处理内部命令简介批处理定义:顾名思义,批处理文件是将一系列命令按一定的顺序集合为一个可执行的文本文件,其扩展名为BAT或者CMD.这些命令统称批处理命令.小知识:可以在 ...