LightOJ 1274 Beating the Dataset(期望)

题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1274

题意:m个1和n-m个0的全排列有SUM=n!/m!/(n-m)!种。对于每一种排列将其看做二进制数x,x右移一位左边补1得到y,将x^y中1的个数称作x的f值。求每个排列的f之和除以SUM。

思路:其实f值就是x中相邻位不同的个数(若x以0开始则再加1,因为它说左边补1),比如f(001)=2,f(101)=2。。。。我们设f[i][j][1]表示前i位含有j个1每次右移左边补1的期望(也就是x以0开始的话加1),f[i][j][0]表示前i位含有j个1每次右移左边补0的期望。则我们有转移方程:

f[i][j][1]=(f[i-1][j][0]+1)*(i-j)/i+f[i-1][j-1][1]*j/i  : f[i-1][j][0]表示左侧补0,补0后还是j个1,所以可以得到 i位j个1,那么f[i-1][j][0]有多少个就对答案贡献多少个f[i-1][j][0]+1,加1是因为f[i][j][1]是左侧补1;f[i-1][j-1][1] 表示左侧补1,得到i为j个1,但是左侧再补1时不再对答案有贡献。那么接下来就是计算f[i-1][j][0]和f[i-1][j-1][1]的个数分别为多少

f[i][j][0]=(f[i-1][j-1][1]+1)*j/i+f[i-1][j][0]*(i-j)/i类似

 

 #include <iostream>
 #include <cstdio>
 #include <cstring>
 #include <algorithm>
 using namespace std;
 
 struct node
 {
     int n,m,id;
	 double ans;
 };
 
 int C;
 double f[2][5005][2];
 node a[15];
 
 int cmp(node a,node b)
 {
     if(a.n!=b.n) return a.n<b.n;
     return a.m<b.m;
 }
 
 int cmp1(node a,node b)
 {
	 return a.id<b.id;
 }
 
 void DP()
 {
     f[0][1][1]=f[0][0][0]=0;
     f[0][1][0]=f[0][0][1]=1;
     int i,j,cur=0,next=1,t=1;
     for(i=1;i<=5000;i++)
     {
         memset(f[next],0,sizeof(f[next]));
         for(j=0;j<=i;j++)
         {
			 while(t<=C&&a[t].n==i&&a[t].m==j)
			 {
				 a[t++].ans=f[cur][j][1];
			 }
             if(t>C) return;
             f[next][j][0]+=f[cur][j][0]*(i+1-j)/(i+1);
             f[next][j][1]+=(f[cur][j][0]+1)*(i+1-j)/(i+1);
             f[next][j+1][0]+=(f[cur][j][1]+1)*(j+1)/(i+1);
             f[next][j+1][1]+=f[cur][j][1]*(j+1)/(i+1);
         }
         cur^=1;
         next^=1;
     }
 }
 
 
 int main()
 {
     scanf("%d",&C);
     int i,n,m;
     for(i=1;i<=C;i++)
     {
         scanf("%d%d",&n,&m);
         a[i].n=n;
         a[i].m=m-n-n;
         a[i].id=i;
     }
     sort(a+1,a+C+1,cmp);
     DP();
	 sort(a+1,a+C+1,cmp1);
     for(i=1;i<=C;i++) printf("Case %d: %.6lf\n",i,a[i].ans);
     return 0;
 }

 

  

 

 

 

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值