【xsy1596】旅行 期望+状压DP

题目大意:有m个人要从城市1开始,依次游览城市1n

每一天,每一个游客有pi的概率去下一个城市,和1pi的概率结束游览。

当游客到达城市j,他会得到(1+CjCj1)Hi,j的收益,其中Ci表示到访第i个城市的人数。

问所有人的期望收益。

数据范围:n,m16

 

我们考虑状压DP

f[i][S]表示到达城市i的人群为S的概率。

ans[i][S]表示到达城市i的人群为S时,所有人在前i座城市的收益和。

不难推出:

f[i][S]=SPf[i1][P]×chg(P,S)

ans[i][S]=SP(ans[i1][P]+f[i1][P]×sum[i][S]×(1+|S||P|))×chg(P,S)

其中,chg(P,S)表示前一次游览的人群为P,下一次剩下S的概率。简单乘一下就行了。

这么转移的复杂度是O(n×3m)的,然而他T了。。。。

复制代码
 1 #include<bits/stdc++.h>
 2 #define M 16
 3 #define get orzmyh
 4 using namespace std;
 5 
 6 int m,n,all=0;
 7 
 8 double p[M+1]={0},h[M+1][M+1]={0},mulp[1<<M]={0},fmulp[1<<M]={0};
 9 double f[M+1][1<<M]={0},ans[M+1][1<<M]={0},mo[1<<M]={0},sum[M+1][1<<M]={0};
10 
11 double Get(int P,int S){
12     return mulp[P&S]*fmulp[P^S];
13 }
14 double Sum(int day,int S){
15     double res=0;
16     for(int i=0;i<m;i++) if((1<<i)&S){
17         res+=h[i][day];
18     }
19     return res;
20 }
21 
22 int main(){
23     scanf("%d%d",&m,&n); all=(1<<m); 
24     for(int i=1;i<all;i++) mo[i]=mo[i>>1]+(i&1);
25     for(int i=0;i<m;i++) scanf("%lf",p+i);
26     
27     for(int i=0;i<all;i++){
28         double mul1=1,mul2=1;
29         for(int j=0;j<m;j++)
30         if(i&(1<<j)){
31             mul1=mul1*p[j];
32             mul2=mul2*(1-p[j]);
33         }
34         mulp[i]=mul1;
35         fmulp[i]=mul2;
36     }
37     
38     for(int i=0;i<m;i++)
39     for(int j=1;j<=n;j++)
40     scanf("%lf",&h[i][j]);
41     
42     for(int i=1;i<=n;i++){
43         for(int S=1;S<all;S++)
44         sum[i][S]=Sum(i,S);
45     }
46     f[1][all-1]=1; ans[1][all-1]=sum[1][all-1];
47     for(int i=2;i<=n;i++){    
48         for(int P=0;P<all;P++)
49         for(int S=P;~S;S=P&(S-1)){
50             double val=Get(S,P);
51             f[i][S]+=f[i-1][P]*val;
52             ans[i][S]+=(ans[i-1][P]+(mo[P]?sum[i][S]*(1+mo[S]/mo[P])*f[i-1][P]:0))*val;
53             if(S==0) break;
54         }
55     }
56     double Ans=0;
57     for(int S=0;S<all;S++)
58     Ans+=ans[n][S];
59     printf("%.10lf\n",Ans);
60 }
复制代码

 

于是我后来写了一个O(nm×2m)的,代码如下:

复制代码
 1 #include<bits/stdc++.h>
 2 #define M 17
 3 using namespace std;
 4 
 5 double f[M][M]={0},h[M][M]={0},p[M]={0};
 6 int n,m;
 7 
 8 int main(){
 9     scanf("%d%d",&m,&n);
10     for(int i=1;i<=m;i++) scanf("%lf",p+i);
11     for(int i=1;i<=m;i++){
12         f[i][1]=1;
13         for(int j=2;j<=n;j++)
14         f[i][j]=f[i][j-1]*p[i];
15     }
16     double ans=0;
17     for(int i=1;i<=m;i++){
18         for(int j=1;j<=n;j++){
19             scanf("%lf",&h[i][j]);
20             ans+=f[i][j]*h[i][j];
21         }
22     }
23     for(int i=1;i<n;i++){
24         for(int s=1;s<(1<<m);s++){
25             double ps=1,sum=0,pn=0; int cnt=0;
26             for(int j=1;j<=m;j++) if(s&(1<<(j-1))){
27                 ps*=f[j][i];
28                 cnt++;
29                 pn+=p[j];
30             }else ps*=1-f[j][i];
31             
32             for(int j=1;j<=m;j++) if(s&(1<<(j-1)))
33             sum+=p[j]*h[j][i+1]*(pn-p[j]+1);
34             ans+=ps*sum/cnt;
35         }
36     }
37     printf("%.10lf\n",ans);
38 }
复制代码
posted @   AlphaInf  阅读(226)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
阅读排行:
· 25岁的心里话
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· 零经验选手,Compose 一天开发一款小游戏!
· 一起来玩mcp_server_sqlite,让AI帮你做增删改查!!
点击右上角即可分享
微信分享提示