\(Maximum\) \(Minimum\) \(identity\)学习笔记

  比较好玩的一个科技。具体来说就是\(max(a,b)=a+b-min(a,b)\),这个式子是比较显然的,但是这个可以扩展到更多数字,比如说\(max(a,b,c)=a+b+c-min(a,b)-min(a,c)-min(b,c)+min(a,b,c)\),其实就是容斥一下。这个东西的用处就是直接求\(max\)比较难求,转化成求\(min\)。

  例题\(1\): \(bzoj\) \(4036\)。

  首先根据\(|\)的性质,如果这一位有了\(1\)就会一直又下去,假设\(t[i]\)表示二进制下第\(i\)位变成\(1\)的时间,那么最后要求的其实就是\(E[max(t[0],t[1],...t[n-1])]\),\(E\)就是期望。这个东西直接求是比较麻烦的,要把问题转化一下,利用\(Maximum\) \(Minimum\) \(identity\),可以把原来的式子化简一下。这里举个简单的例子,比如说只有两位数,那么\(E[max(t[0],t[1])]=E[t[0]]+E[t[1]]-E[min(t[0],t[1])]\),如何求\(min(t[0],t[1])\)呢?其实就是相当于第\(0\)位与第\(1\)位均没有变成\(1\)的期望时间,设集合\(S\)为全集减去\(0,1\)的集合,操作集合就是\(S\)的一个子集。然后根据公式 期望时间\(T=1/p\),\(p\)为概率。只需要y用\(FMT\)把一个集合的所有子集概率和算出来就行了,最后枚举一下状态,看一下\(1\)的个数来确定容斥系数,时间复杂度\(O(2^n*n)\)。

#include<iostream>
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<cstdlib> using namespace std;
const int MAXN = (1<<20)+5; int n;
double p[MAXN],ans; int main(){
scanf("%d",&n);
for(int i=0;i<(1<<n);i++) scanf("%lf",&p[i]);
for(int i=1;i<=n;i++)
for(int j=0;j<(1<<n);j++)
if((j&(1<<(i-1)))) p[j]+=p[j^(1<<(i-1))];
for(int i=0;i<(1<<n)-1;i++){
if(p[i]>=1.0-1e-7) {puts("INF");return 0;}
int now=__builtin_popcount(i);now=n-now;
if(now&1) ans+=1.0/(1.0-p[i]);
else ans-=1.0/(1.0-p[i]);
}
printf("%.10lf",ans);
return 0;
}

  例题\(2\):\(hdu\) \(4624\)

  题目相当于给一个序列染色,首先可以利用上一道题的思路直接算,但时间复杂度是\(O(2^n)\)的。需要一个多项式时间复杂度的算法,自然而然能想到\(dp\)。我们把这个序列是否被染过色用一个\(0/1\)串表示,如果被染过则为\(1\)。我们发现其实需要的信息只有两条,一个是染过的奇偶性,这个用来确定容斥系数,还有一个是有多少个操作能操作到没被染过颜色的集合。设\(f[i][0/1][j]\)表示染到第\(i\)位,\(0/1\)表示为奇数还是偶数,\(j\)表示能操作到的没被染过色的集合。那么转移方程为\(f[i][j][k]->f[l][j\) \(xor\) \(1][k+C_{l-i-1}^2 ]\)。首先预处理出\(f\)数组,然后就可以计算期望\(E[x]\),这就与上一题的计算方式差不多了,只需要枚举每一个状态,注意还要算上\(x\)这个点到枚举的终点这一段的区间。具体看代码,\(hdu\)需要保留\(15\)位小数,必须要写高精度,我懒得写了。。。时间复杂度\(O(n^4)\)。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib> using namespace std;
const int MAXN = 55;
const double eps = 1e-8;
typedef long long LL; inline int rd(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)) {f=ch=='-'?0:1;ch=getchar();}
while(isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return f?x:-x;
} LL f[MAXN][2][MAXN*MAXN/2];
int n,T;
double E[MAXN]; int main(){
f[0][0][0]=1;
for(int i=0;i<=49;i++)
for(int j=0;j<=(i*(i+1))/2;j++)
for(int op=0;op<=1;op++)if(f[i][op][j])
for(int k=i+1;k<=50;k++)
f[k][op^1][j+(k-i)*(k-i-1)/2]+=f[i][op][j];
for(int i=1;i<=50;i++)
for(int j=0;j<=i;j++)
for(int k=0;k<=(j+1)*j/2;k++)
for(int op=0;op<=1;op++)if(f[j][op][k]){
double p=(double)(k+(i-j)*(i-j+1)/2)/(i*(i+1)/2);
if(p>=1.0-eps) continue;
if(op) E[i]+=(double)f[j][op][k]/(1-p);
else E[i]-=(double)f[j][op][k]/(1-p);
}
T=rd();
while(T--){
n=rd();printf("%.15lf\n",E[n]);
}
return 0;
}

随机推荐

  1. sql update语句

    如果要更新数据库表中的记录,我们就必须使用UPDATE语句. UPDATE语句的基本语法是: UPDATE <表名> SET 字段1=值1, 字段2=值2, ... WHERE ...; ...

  2. Linux下安装PHP的curl扩展

    先安装依赖包: yum install curl curl-devel 找到PHP的安装包,cd 进入安装包 cd php-5.6.25/ext/curl phpize 如果报找不到phpize就补全 ...

  3. 64 计算机图形学入门(1)——OpenGL环境配置与图形流水线(图像管线)

    0 引言 最近想学一下计算机图形学方面的知识,原因如下.目前本人接触了数字图像处理(opencv)以及点云处理(PCL)方面的知识,对从图像和点云中提取特征信息,并将特征转化为底层/中层语义信息有了一 ...

  4. 详解Windows注册表分析取证

    大多数都知道windows系统中有个叫注册表的东西,但却很少有人会去深入的了解它的作用以及如何对它进行操作.然而对于计算机取证人员来说注册表无疑是块巨大的宝藏.通过注册表取证人员能分析出系统发生了什么 ...

  5. SDNU 1217 CD收藏——并查集

    Description     lmh平常爱听歌,所以买了很多的CD来收藏,但是因为平常整理不当,所以忘记了这些CD的歌手是谁.现在他想知道他到底收藏了多少位歌手的专辑,于是他想了一个办法,同时拿出两 ...

  6. A. Srdce and Triangle--“今日头条杯”首届湖北省大学程序设计竞赛(网络同步赛)

    如下图这是“今日头条杯”首届湖北省大学程序设计竞赛的第一题,作为赛后补题 题目描述:链接点此 这套题的github地址(里面包含了数据,题解,现场排名):点此 Let  be a regualr tr ...

  7. Robotframework之下拉列表select

    下拉框控件很常见啊,主要说一下robotframework中怎么玩转下拉框,第一点要注意的就是,别看到下拉的就用select控件去操作,因为很多下拉列表用的不一定就是select控件.robotfra ...

  8. 三(1)、springcloud之Eureka服务注册与发现

    1.认识Eureka ​ Eureka是Netflix的一个子模块,也是核心模块之一.Eureka是一个基于REST的服务,用于定位服务,以实现云端中间层服务发现和故障转移.服务注册与发现对于微服务架 ...

  9. python-Request模块使用

    request 允许发送HTTP/1.1的请求,并为我们封装了更多的方法让我们不需要手动为 URL 添加查询字串,也不需要对 POST 数据进行表单编码.Keep-alive 和 HTTP 连接池的功 ...

  10. 注册页面-使用form模块搭建

    基于Django的form模块,快速的搭建注册页面,每个限制条件,都放在form模块里面,不单独对每一项编写标签,使用模版的 for 循环来渲染. 首先设置form模块 在blogs模块下创建一个bl ...