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 ...
随机推荐
- IE 浏览器各个版本 JavaScript 支持情况一览表
语言元素 语言元素 突发.IE6 标准.IE7 标准 IE8 标准 IE 9 标准 IE 10 标准 边缘 Windows 应用商店应用程序 __proto__ 属性 (Object) (JavaSc ...
- 安装Oracle时出现环境变量Path的值大于1023的解决办法
出现的情况我就不说了,直接重点: 计算机->属性->高级系统设置->高级->环境变量 1)在"系统变量"编辑Path,全选将其中的路径全部复制出来放到文本文 ...
- AngularJs最简单解决跨域问题案例
AngularJs最简单解决跨域问题案例 2016-05-20 09:18 82人阅读 评论(0) 收藏 举报 分类: javascript(1) 作者:白狼 出处:http://www.mank ...
- HTTP、HTTP2
HTTP.HTTP2.0.SPDY.HTTPS 你应该知道的一些事 原文链接:http://www.alloyteam.com/2016/07/httphttp2-0spdyhttps-readi ...
- Laravel大型项目系列教程(四)显示文章列表和用户修改文章
小编心语:不知不觉已经第四部分了,非常感谢很多人给小编提的意见,改了很多bug,希望以后能继续帮小编找找茬~小编也不希望误导大家~这一节,主要讲的 是如何显示文章列表和让用户修改文章,小编预告一下(一 ...
- MVC学习系列4--@helper辅助方法和用户自定义HTML方法
在HTML Helper,帮助类的帮助下,我们可以动态的创建HTML控件.HTML帮助类是在视图中,用来呈现HTML内容的.HTML帮助类是一个方法,它返回的是string类型的值. HTML帮助类, ...
- 使用c/c++扩展python
用python脚本写应用比较方便,但有时候由于种种原因需要扩展python(比如给程序提供python接口等). 之前一直想整理下,今天终于坐下来把这件事情给做了,这里记录下,也方便我以后查阅. 说明 ...
- 论Pair的重要性
这些天我在用React和D3做图表,从已经实现的图表里复制了一些坐标轴的代码,发现坐标轴上的n个点里,只有第一个点下面能渲染出文字提示,其余点下面都无法渲染出文字. 和组里的FL一起百思不得其解好几天 ...
- GreenPlum高效去除表重复数据
1.针对PostgreSQL数据库表的去重复方法基本有三种,这是在网上查找的方法,在附录1给出.但是这些方法对GreenPlum来说都不管用. 2.数据表分布在不同的节点上,每个节点的ctid是唯一的 ...
- My first win32 application program
#include<afxwin.h>#include<afx.h>#define _AFXDLLclass CHelloApp :public CWinApp{public: ...