zoj 3329 One Person Game 概率DP

思路:这题的递推方程有点麻烦!!

dp[i]表示分数为i的期望步数,p[k]表示得分为k的概率,p0表示回到0的概率:

dp[i]=Σ(p[k]*dp[i+k])+dp[0]*p0+1

设dp[i]=A[i]*dp[0]+B[i]带入的:

dp[i]=∑(pk*A[i+k]*dp[0]+pk*B[i+k])+dp[0]*p0+1
       =(∑(pk*A[i+k])+p0)dp[0]+∑(pk*B[i+k])+1;
     明显A[i]=(∑(pk*A[i+k])+p0)
     B[i]=∑(pk*B[i+k])+1
     先递推求得A[0]和B[0].
     那么  dp[0]=B[0]/(1-A[0]);

链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=3754

代码如下:

 

 1 #include<iostream>

 2 #include<stdio.h>

 3 #include<algorithm>

 4 #include<iomanip>

 5 #include<cmath>

 6 #include<cstring>

 7 #include<vector>

 8 #define ll __int64

 9 #define pi acos(-1.0)

10 #define MAX 600

11 using namespace std;

12 double p[100],A[MAX],B[MAX],p0;

13 int main(){

14     int n,i,j,t,k,s,k1,k2,k3,a,b,c;

15     cin>>t;

16     while(t--){

17         cin>>n>>k1>>k2>>k3>>a>>b>>c;

18         p0=1.0/k1/k2/k3;

19         memset(p,0,sizeof(p));

20         for(i=1;i<=k1;i++)

21         for(j=1;j<=k2;j++)

22         for(k=1;k<=k3;k++){

23             if(i!=a||j!=b||k!=c)

24                 p[i+j+k]+=p0;

25         }

26         memset(A,0,sizeof(A));

27         memset(B,0,sizeof(B));

28         for(i=n;i>=0;i--){

29             A[i]=p0;B[i]=1;

30             for(j=1;j<=k1+k2+k3;j++){

31                 A[i]+=A[i+j]*p[j];

32                 B[i]+=B[i+j]*p[j];

33             }

34         }

35         printf("%.15lf\n",B[0]/(1.0-A[0]));

36     }

37     return 0;

38 }
View Code

 

 

 

你可能感兴趣的:(game)