题目:
题意:有s个系统,有n种bug,bug的数量不限,一位程序员每天可以发现一个bug
现在求发现n种bug存在s个系统中并且每个系统都要被发现bug的平均天数(期望)
题解:
看来能化的就要化到最简,这样就少一些精度错误
虽然不太清楚为什么,可能是套路吧做多了就好了
在这里有s,n两个变量,假设dp[i][j]表示已经发现i种bug存在j个系统到目标所需要的天数,显然dp[n][s]=0;
从dp[i][j]经过一天后可能得到以下四种情况:
dp[i][j]->dp[i+1][j+1];//在一个新的系统里面发现一个新的bug
dp[i][j]->dp[i+1][j];//在原来已发现过bug的系统里发现一个新的bug
dp[i][j]->dp[i][j+1];//在一个新的系统里面发现一个已被发现过的bug
dp[i][j]->dp[i][j];//在已发现过bug的系统发现已发现过的bug
四种到达的概率分别是:
p1=(n-i)*(s-j)/(n*s);
p2=(n-i)*j/(n*s);
p3=i*(s-j)/(n*s);
p4=i*j/(n*s);
然后根据E(aA+bB+cC+dD+...)=aEA+bEB+....;//a,b,c,d...表示概率,A,B,C...表示状态
所以dp[i][j]=p1*dp[i+1][j+1]+p2*dp[i+1][j]+p3*dp[i][j+1]+p4*dp[i][j]+1;//dp[i][j]表示的就是到达状态i,j的期望
=>dp[i][j]=(p1*dp[i+1][j+1]+p2*dp[i+1][j]+p3*dp[i][j+1]+1)/(1-p4);
代码:
#include <cstdio>
using namespace std;
double dp[1005][1005];
int main()
{
int n,s,i,j;
scanf("%d%d",&n,&s);//n个bug,s个系统
dp[n][s]=0;int ns=n*s;
for (i=n;i>=0;i--)//dp[i][j]表示已经找到i种bug,j个系统的期望天数
for (j=s;j>=0;j--)
{
if (i==n && j==s) continue;
//dp[i][j]=((dp[i][j]*(i*j)/(n*s)+dp[i][j+1]*(i*(s-j))/(n*s)+dp[i+1][j]*((n-i)*j)/(n*s)+dp[i+1][j+1]*((n-i)*(s-j))/(n*s)))/((1.0-(i*j)/(n*s)));最开始的式子
//dp[i][j]=((1.0+dp[i][j+1]*(i*(s-j))/(n*s)+dp[i+1][j]*((n-i)*j)/(n*s)+dp[i+1][j+1]*((n-i)*(s-j))/(n*s)))/((1.0-(i*j)/(n*s))); 对这种就是错的,虽然和下面的式子一样,但是/n*s毕竟不是准确的
dp[i][j]=(ns+dp[i][j+1]*(i*(s-j))+dp[i+1][j]*((n-i)*j)+dp[i+1][j+1]*((n-i)*(s-j)))/(n*s-i*j);
}
printf("%.4lf",dp[0][0]);
}