[HAOI2015]按位或
题解
按照惯例,我们先进行拆位,每一个二进制位分开考虑,假设第 i i i 位第一次变为1的时间是 f ( i ) f(i) f(i),那么答案就是 E ( max { f ( i ) } ) E(\max\{f(i)\}) E(max{f(i)})。
这么
max
\max
max 符号在期望的里面,不能直接拿出来,所以最好的处理办法就是对它使用 Min-Max 反演:
E
(
max
i
∈
S
{
f
(
i
)
}
)
=
∑
T
≠
∅
,
T
⊆
S
(
−
1
)
∣
T
∣
−
1
E
(
min
i
∈
T
{
f
(
i
)
}
)
E(\max_{i\in S}\{f(i)\})=\sum_{T\neq ∅,T\subseteq S}(-1)^{|T|-1}E(\min_{i\in T}\{f(i)\})
E(i∈Smax{f(i)})=T=∅,T⊆S∑(−1)∣T∣−1E(i∈Tmin{f(i)})
反演一下就简单多了,我们只需要求出每个集合内第一次出现1的期望时间即可。设
s
p
S
=
∑
T
⊆
S
p
T
sp_S=\sum_{T\subseteq S}p_T
spS=∑T⊆SpT,那么有
E
(
min
i
∈
S
{
f
(
i
)
}
)
=
s
p
I
−
S
1
−
s
p
I
−
S
+
1
=
1
1
−
s
p
I
−
S
E(\min_{i\in S}\{f(i)\})=\frac{sp_{I-S}}{1-sp_{I-S}}+1=\frac{1}{1-sp_{I-S}}
E(i∈Smin{f(i)})=1−spI−SspI−S+1=1−spI−S1
这个
s
p
sp
sp 用沃尔什变换求一下即可。
代码
今天发现一个巨坑,就是主函数想要用逗号执行一步操作后返回值,如果忘了在后面加,0
并且前面部分返回的值正好是int则不会报错,但是运行时会RE
。
#include<bits/stdc++.h>//JZM yyds!!
#define ll long long
#define uns unsigned
#define IF (it->first)
#define IS (it->second)
#define END putchar('\n')
using namespace std;
const int MAXN=1048581;
const ll INF=1e18;
inline ll read(){
ll x=0;bool f=1;char s=getchar();
while((s<'0'||s>'9')&&s>0){if(s=='-')f^=1;s=getchar();}
while(s>='0'&&s<='9')x=(x<<1)+(x<<3)+(s^48),s=getchar();
return f?x:-x;
}
int ptf[50],lpt;
inline void print(ll x,char c='\n'){
if(x<0)putchar('-'),x=-x;
ptf[lpt=1]=x%10;
while(x>9)x/=10,ptf[++lpt]=x%10;
while(lpt)putchar(ptf[lpt--]^48);
if(c>0)putchar(c);
}
inline ll lowbit(ll x){return x&-x;}
int n,g[MAXN];
const double eps=1e-9;
double p[MAXN],ans;
signed main()
{
n=read();
for(int s=0;s<(1<<n);s++)scanf("%lf",p+s);
for(int k=0;k<n;k++)
for(int s=0;s<(1<<n);s++)
if((s>>k)&1)p[s]+=p[s^(1<<k)];
g[0]=-1;
for(int s=1;s<(1<<n);s++)g[s]=(s&1)?-g[s>>1]:g[s>>1];
for(int s=1,lim=(1<<n)-1;s<=lim;s++){
if(1-p[lim^s]<eps)return printf("INF\n"),0;//就是这里
ans+=1.0*g[s]/(1-p[lim^s]);
}
printf("%.10f\n",ans);
return 0;
}