POJ2151-Check the difficulty of problems

全解题报告索引目录 -> 【北大ACM – POJ试题分类

转载请注明出处:http://exp-blog.com

-------------------------------------------------------------------------

 

 

 

大致题意:

ACM比赛中,共M道题,T个队,pij表示第i队解出第j题的概率

问 每队至少解出一题且冠军队至少解出N道题的概率。

 

解题思路:

真费解为什么这题被划分到了Hash。。。

明明是 概率+DP  ,概率不好真的拿不下这题T .T,

这题难点不在编程,在于问题的转化和理解= =只要能用笔算出答案,离AC也就不远了。。。

 

要求:每队至少解出一题 且 冠军队至少解出N道题的概率

由于冠军队可以不止一队,即允许存在并列冠军

则原来的所求的概率可以转化为:

每队均至少做一题的概率P1 减去 每队做题数均在1到N-1之间的概率P2

 

 

 

 

//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;
}


 

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值