好久之前学过,记得是一次亚洲区的前几天看了看概率dp,然后亚洲区就出了一道概率dp,当时虽然做上了,但是感觉有很多地方没懂,今天起早温习了一下,觉得很多地方茅塞顿开,果然学习的话早上效果最好了。

首先来看一道最基础概率dp

题意是,有一个软件,有s个子系统,会产生n种bug。 
某个程序员一天能发现一个bug,这个bug是这n种bug中的一种,然后发生在某个子系统中。 
问,找到所有的n种bug,且每个子系统都找到bug,这样所要的天数,的期望。

期望,可以分解成多个子期望的加权和,权为子期望发生的概率 
所以: 我首先想到了一个这样的公式dp[x][y] = dp[x][y]*p1+dp[x-1][y-1]*p2+dp[x][y-1]*p3+dp[x-1][y]*p4

dp[x][y]代表已经有x种bug并且有y个系统至少有一个bug的期望值

我们知道他是从自身以及他的前几种状态推导过来,乍一看这个公式应该是对的,但是dp[n][m]会无穷大,因为他的期望是没有停止状态的,也就是题意要求的应该是到达n,m状态时停止。

那么我们又可以从倒推的角度去考虑这个问题,dp[x][y]表示的是已经有x种bug并且有y个系统至少有一种bug的时候还需要多少步能够到达dp[n][m]的状态。

那么公式又变成了这样dp[x][y] = dp[x+1][y+1]*p1+dp[x+1][y]*p2+dp[x][y+1]*p3+dp[x][y]*p4+1

也就是dp[x][y] = dp[x+1][y+1] *(n-x)*(m-y)/n/m+dp[x+1][y]*(n-x)*y/n/m+dp[x][y+1]*x*(m-y)/n/m+dp[x][y]*x*y/n/m+1

将dp[x][y]合并得dp[x][y](1-x*y/n/m) = dp[x+1][y+1] *(n-x)*(m-y)/n/m+dp[x+1][y]*(n-x)*y/n/m+dp[x][y+1]*x*(m-y)/n/m+1

1.poj2096,就是上面的题

#include<stdio.h>
#include<string.h>
#include<iostream>
using namespace std;
double dp[][];
int main(){
int n, m;
scanf("%d%d", &n, &m);
for(int i = ;i <= n+; i++){
for(int k = ; k <= m+; k++){
dp[i][k]= ;
}
}
for(int x = n; x >= ; x--){
for(int y = m; y >=; y--){
if(x == n && y ==m)dp[x][y] = ;
else
dp[x][y]= (+dp[x+][y+] *(n-x)*(m-y)/n/m+dp[x+][y]*(n-x)*y/n/m+dp[x][y+]*x*(m-y)/n/m)/(1.0-1.0*x*y/n/m);
}
}
printf("%.4f\n", dp[][]);
}

2.UVALive 5811

题目大意:有54张牌,一张一张翻,在四种花色都不小于分别给定的四个值时停止,王牌可以当做任意一种花色,但是在翻得时候就要确定,求翻牌数的期望。

公式神马的好推,写的时候稍微费劲了点

#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
double dp[][][][][][];
int main(){
int t;
scanf("%d", &t);
int cas = ;
while(t--){
int a, b, c, d;
scanf("%d%d%d%d", &a, &b, &c, &d);
int aa = ;
if(a > ) aa += -a;
if(b > ) aa += -b;
if(c > ) aa += - c;
if(d > ) aa += - d;
if(aa < -){
printf("Case %d: %.3f\n", cas++, -1.0);//printf("000");
continue;
}
for(int i = ; i >= ; i--){
for(int k = ; k >= ;k --){
for(int j = ; j >= ; j--){
for(int h = ; h >= ; h--){
for(int x = ; x >= ; x--){
for(int y = ; y >= ; y--){
int aa = i, bb = k, cc = j, dd = h;
if(x == ) aa++;
if(x == ) bb ++;
if(x == ) cc++;
if(x == ) dd++;
if(y == ) aa++;
if(y==) bb++;
if(y == ) cc++;
if(y == ) dd++;
int tot = - aa - bb - cc- dd;
if(aa >= a && bb >= b && cc >= c && dd >= d){
dp[i][k][j][h][x][y] = ;continue;
}else if(tot == ){
dp[i][k][j][h][x][y] = ;continue;
}
double uu = ;
int ss = ; //kexuanwangpai
if(!x) ss ++;
if(!y) ss ++;
dp[i][k][j][h][x][y] = ;
if(!x){
uu = min(min(min(dp[i][k][j][h][][y], dp[i][k][j][h][][y]), dp[i][k][j][h][][y]), dp[i][k][j][h][][y]);
//if(i+k+j+h == 52 && x == 0 && y == 0) printf("%.3lf %.3lf-----------\n", uu,ss*1.0/tot*uu);
dp[i][k][j][h][x][y] += ss*1.0/tot*uu;
}
else if(!y){
uu = min(min(min(dp[i][k][j][h][x][],dp[i][k][j][h][x][]),dp[i][k][j][h][x][]), dp[i][k][j][h][x][]);
dp[i][k][j][h][x][y] += ss * 1.0/tot*uu;
}
uu = ;
if(i < ){
uu += dp[i+][k][j][h][x][y]*(-i)/tot;
}
if(k < )
uu +=dp[i][k+][j][h][x][y] * (-k)/tot;
if(j < )
uu += dp[i][k][j+][h][x][y] *( - j)/tot;
if(h < )
uu += dp[i][k][j][h+][x][y] * ( - h)/tot;
dp[i][k][j][h][x][y] += uu;
}
}
}
}
}
}
//printf("%.3lf\n", dp[13][13][13][13][0][0]);
printf("Case %d: %.3f\n", cas++, dp[][][][][][]);
}
}

概率dp小结的更多相关文章

  1. 概率及期望DP小结

    资源分享 26 个比较概率大小的问题 数论小白都能看懂的数学期望讲解 概念 \(PS\):不需要知道太多概念,能拿来用就行了. 定义 样本(\(\omega\)):一次随机试验产生的一个结果. 样本空 ...

  2. Codeforces 28C [概率DP]

    /* 大连热身D题 题意: 有n个人,m个浴室每个浴室有ai个喷头,每个人等概率得选择一个浴室. 每个浴室的人都在喷头前边排队,而且每个浴室内保证大家都尽可能均匀得在喷头后边排队. 求所有浴室中最长队 ...

  3. HDU 4405 Aeroplane chess (概率DP)

    题意:你从0开始,要跳到 n 这个位置,如果当前位置是一个飞行点,那么可以跳过去,要不然就只能掷骰子,问你要掷的次数数学期望,到达或者超过n. 析:概率DP,dp[i] 表示从 i  这个位置到达 n ...

  4. POJ 2096 Collecting Bugs (概率DP)

    题意:给定 n 类bug,和 s 个子系统,每天可以找出一个bug,求找出 n 类型的bug,并且 s 个都至少有一个的期望是多少. 析:应该是一个很简单的概率DP,dp[i][j] 表示已经从 j ...

  5. POJ 2151 Check the difficulty of problems (概率DP)

    题意:ACM比赛中,共M道题,T个队,pij表示第i队解出第j题的概率 ,求每队至少解出一题且冠军队至少解出N道题的概率. 析:概率DP,dp[i][j][k] 表示第 i 个队伍,前 j 个题,解出 ...

  6. 概率DP light oj 1030

    t组数据 n块黄金 到这里就捡起来 出发点1 到n结束  点+位置>n 重掷一次 dp[i] 代表到这里的概率 dp[i]=(dp[i-1]+dp[i-2]... )/6  如果满6个的话 否则 ...

  7. hdu 4050 2011北京赛区网络赛K 概率dp ***

    题目:给出1-n连续的方格,从0开始,每一个格子有4个状态,左右脚交替,向右跳,而且每一步的步长必须在给定的区间之内.当跳出n个格子或者没有格子可以跳的时候就结束了,求出游戏的期望步数 0:表示不能到 ...

  8. [转]概率DP总结 by kuangbin

    概率类题目一直比较弱,准备把kuangbin大师傅总结的这篇题刷一下! 我把下面的代码换成了自己的代码! 原文地址:http://www.cnblogs.com/kuangbin/archive/20 ...

  9. SGU 422 Fast Typing(概率DP)

    题目大意 某人在打字机上打一个字符串,给出了他打每个字符出错的概率 q[i]. 打一个字符需要单位1的时间,删除一个字符也需要单位1的时间.在任意时刻,他可以花 t 的时间检查整个打出来的字符串,并且 ...

随机推荐

  1. phpcms v9 读取地区联动菜单缓存文件

    读取缓存文件的方法是 getcache() 在 phpcms\libs\functions\global.func.php 中可找到. 地区联动菜单的缓存文件是  caches\caches_link ...

  2. xheditor在线编辑器的使用

    在你所需要在线编辑器的工程目录下,导入xheditor_emot.xheditor_plugins和xheditor_skin.jquery四个文件夹,然后在textarea标签中加入: class= ...

  3. 数值统计 AC 杭电

    数值统计 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  4. 对编程语言的需求总结为四个:效率,灵活,抽象,生产率(C++玩的是前三个,Java和C#玩的是后两个)

    Why C++ ? 王者归来(转载) 因为又有人邀请我去Quora的C2C网站去回答问题去了,这回是 关于 @laiyonghao 的这篇有点争议的博文<2012 不宜进入的三个技术点>A ...

  5. 用QtWebKit开发简单的浏览器

    用QtWebKit开发简单的浏览器 1.代码实现 工程目录结构如下: AddressBar类包含了地址栏和按钮两个控件,将地址栏回车和按钮点击信号与goToSite()槽连接. 当回车和点击事件发生时 ...

  6. SPOJ 0287 Smart Network Administrator

    题目大意:一座村庄有N户人家.只有第一家可以连上互联网,其他人家要想上网必须拉一根缆线通过若干条街道连到第一家.每一根完整的缆线只能有一种颜色.网管有一个要求,各条街道内不同人家的缆线必须不同色,且总 ...

  7. HDOJ 1303 Doubles(简单题)

    Problem Description As part of an arithmetic competency program, your students will be given randoml ...

  8. Codeforces Round #203 - D. Looking for Owls

    D. Looking for Owls Emperor Palpatine loves owls very much. The emperor has some blueprints with the ...

  9. vmware vms migration to openstack

    Converting a VMware Workstation virtual machine to KVM Leave a commentPosted by rbgeek on August 13, ...

  10. C#调用PowerShell脚本

    今天通过一个小例子,学习了C#如何调用PowerShell脚本文件的Function以及传参. private bool CallPowershell(string outputFile) { str ...