- POJ 2151 - Check the difficulty of problems
- Time: 2000MS
- Memory: 65536K
- 难度: 初级
- 分类: 概率
问题描述
ACM比赛中,共M道题,T个队,pij表示第i队解出第j题的概率
问 每队至少解出一题且冠军队至少解出N道题的概率。
解题思路
概率+DP ,概率不好真的拿不下这题
这题难点不在编程,在于问题的转化和理解
只要能用笔算出答案,离AC也就不远了。。。
要求:
- 每队至少解出一题 且 冠军队至少解出N道题的概率
- 由于冠军队可以不止一队,即允许存在并列冠军
则原来的所求的概率可以转化为:
每队均至少做一题的概率P1 减去 每队做题数均在1到N-1之间的概率P2
AC 源码
//Memory Time
//8272K 110MS
#include<iostream>
#include<iomanip>
using namespace std;
int M,T,N; //M:题数 T:队数 N:冠军队至少做题数
double dp[1001][31][31]; //状态方程: dp[i][j][k]为第i队PASS前j题中的k题的概率
double p[1001][31]; //p[i][j]为第i队通过第j题的概率
double s[1001][31]; //s[i][j]为第i队在M题中至少PASS j题的概率
void ProTable(void) //概率打表
{
memset(dp,0.0,sizeof(dp));
memset(s,0.0,sizeof(s));
int i,j,k;
for(i=1;i<=T;i++) //逐队枚举
{
/*Initial*/
dp[i][0][0]=1.0;
for(j=1;j<=M;j++)
dp[i][j][0]=dp[i][j-1][0]*(1-p[i][j]);
/*Dp*/
for(j=1;j<=M;j++)
for(k=1;k<=j;k++)
dp[i][j][k] = dp[i][j-1][k-1]*p[i][j] + dp[i][j-1][k]*(1-p[i][j]);
s[i][0]=dp[i][M][0];
for(k=1;k<=M;k++)
s[i][k]=s[i][k-1]+dp[i][M][k];
}
return;
}
int main(int i,int j)
{
while(cin>>M>>T>>N)
{
if(!M && !T && !N)
break;
/*Input*/
for(i=1;i<=T;i++)
for(j=1;j<=M;j++)
cin>>p[i][j];
/*Compute the Probability*/
ProTable();
double p1=1.0;
for(i=1;i<=T;i++)
p1*=(s[i][M]-s[i][0]); //所有队至少做1题的概率
double p2=1.0;
for(i=1;i<=T;i++)
p2*=(s[i][N-1]-s[i][0]); //所有队做的题数均在1~N-1之间的概率
/*Output*/
cout<<fixed<<setprecision(3)<<p1-p2<<endl;
//每队至少解出一题 且 至少有一队(冠军队)能至少解出N道题的概率
}
return 0;
}