\(\\\)

Description


给出一个数列,多次询问,每次给出一个区间 \([l_i,r_i]\) ,问在区间中随意选两个位置不同的数,是同一个数的概率有多大。

  • \(n,m\le 5\times 10^4\)

\(\\\)

Solution


莫队模板。

才发现自己还没有正经学过莫队

设 \(cnt[i]\) 表示 \(i\) 在 \([l,r]\) 中出现的个数,\(S\) 为这个区间的数集,一个区间的答案可以表示成:

\[\frac{\sum_{x\in S}C_{cnt[x]}^2}{C_{r-l+1}^2}
\]

分母好说,考虑如何求分子,不妨展开一下。

\[\sum_{x\in S}C_{cnt[x]}^2=\sum_{x\in S}\frac{cnt[x]\times(cnt[x]-1)}{2}=\frac 12\times\bigg( \sum_{x\in S}cnt[x]^2-\sum_{x\in S}cnt[x]\bigg)
\]

于是有一个机智的做法:我们设 \(ans\) 表示括号里面的东西。

每次区间移动时,假如 \(cnt[x]\to cnt[x]'\) ,那么对 \(ans\) 的更新有:

\[ans=ans-cnt[x]^2+cnt[x]+cnt[x]'^2-cnt[x]'
\]

就可以 \(O(1)\) 移动回答区间的左端点或右端点了。

关于询问的排序:左端点在同一个块内的按照右端点排序,不在同一个块内的按照左端点所在块排序。

这样每个块右端点至多移动 \(O(N)\),左端点至多移动 \(O(q\sqrt N)\) ,其中 \(q\) 是左端点在这个块内的询问个数。

总复杂度 \(O(N\sqrt N)\)

\(\\\)

Code


#include<cmath>
#include<cstdio>
#include<cctype>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 50010
#define R register
#define gc getchar
using namespace std;
typedef long long ll; inline ll rd(){
ll x=0; char c=gc();
while(!isdigit(c)) c=gc();
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=gc();}
return x;
} ll ans,n,m,s[N],cnt[N],bl[N]; struct Q{ll l,r,id,ans1,ans2;}q[N]; inline bool cmp(Q x,Q y){
return bl[x.l]==bl[y.l]?x.r<y.r:bl[x.l]<bl[y.l];
} inline bool cmp1(Q x,Q y){return x.id<y.id;} inline ll C(ll x){return (x*(x-1)>>1);} ll gcd(ll x,ll y){return y?gcd(y,x%y):x;} inline void move(ll &llst,ll &rlst,ll lnew,ll rnew){
while(llst<lnew){
int x=cnt[s[llst]];
ans=ans-x*x+(x-1)*(x-1)+1;
--cnt[s[llst]]; ++llst;
}
while(llst>lnew){
--llst;
int x=cnt[s[llst]];
ans=ans-x*x+(x+1)*(x+1)-1;
++cnt[s[llst]];
}
while(rlst<rnew){
++rlst;
int x=cnt[s[rlst]];
ans=ans-x*x+(x+1)*(x+1)-1;
++cnt[s[rlst]];
}
while(rlst>rnew){
int x=cnt[s[rlst]];
ans=ans-x*x+(x-1)*(x-1)+1;
--cnt[s[rlst]]; --rlst;
}
} int main(){
ll n=rd(),m=rd();
ll t=sqrt(n);
for(R ll i=1,cntt=1;i<=n;++i){
if(i%t==0) ++cntt;
bl[i]=cntt;
}
for(R ll i=1;i<=n;++i) s[i]=rd();
for(R ll i=1;i<=m;++i){q[i].l=rd();q[i].r=rd();q[i].id=i;}
sort(q+1,q+1+m,cmp);
ll lnow=1,rnow=1;
++cnt[s[1]];
for(R ll i=1,g;i<=m;++i){
if(q[i].l==q[i].r){
q[i].ans1=0;
q[i].ans2=1;
continue;
}
move(lnow,rnow,q[i].l,q[i].r);
g=gcd(ans/2,C(q[i].r-q[i].l+1));
q[i].ans1=ans/g/2;
q[i].ans2=C(q[i].r-q[i].l+1)/g;
}
sort(q+1,q+1+m,cmp1);
for(R int i=1;i<=m;++i) printf("%lld/%lld\n",q[i].ans1,q[i].ans2);
return 0;
}

[ BZOJ 2038 ] Hose的更多相关文章

  1. 【莫队】bzoj 3781,bzoj 2038,bzoj 3289

    好像又有一个星期没更博客了.. 最近疯狂考试...唯一有点收获的就是学会了莫队这种神奇的算法.. 听起来很难..其实是一个很简单的东西.. 就是在区间处理问题时对于一个待求区间[L',R']通过之前求 ...

  2. [BZOJ 2038]小Z的袜子

    传送门:BZOJ 2038 题意很明确,是在给定的区间内任意选取两个数,求选到两个相同的数的概率. 所以我们得首先统计在给定的区间内,相同的数对有多少对,那么这里就使用到了莫队算法.如果对莫队算法还不 ...

  3. BZOJ 2038: [2009国家集训队]小Z的袜子

    二次联通门 : BZOJ 2038: [2009国家集训队]小Z的袜子 /* BZOJ 2038: [2009国家集训队]小Z的袜子 莫队经典题 但是我并不认为此题适合入门.. Answer = ∑ ...

  4. BZOJ 2038: [2009国家集训队]小Z的袜子(hose) [莫队算法]【学习笔记】

    2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 7687  Solved: 3516[Subm ...

  5. BZOJ 2038: [2009国家集训队]小Z的袜子(hose)

    2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 7676  Solved: 3509[Subm ...

  6. Bzoj 2038: [2009国家集训队]小Z的袜子(hose) 莫队,分块,暴力

    2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 5763  Solved: 2660[Subm ...

  7. BZOJ 2038: [2009国家集训队]小Z的袜子(hose) ( 莫队 )

    莫队..先按sqrt(n)分块, 然后按块的顺序对询问排序, 同块就按右端点排序. 然后就按排序后的顺序暴力求解即可. 时间复杂度O(n1.5) --------------------------- ...

  8. BZOJ 2038: [2009国家集训队]小Z的袜子(hose) 分块

    分块大法好 2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec  Memory Limit: 259 MB Submit: 2938  Solved: 13 ...

  9. BZOJ 2038: [2009国家集训队]小Z的袜子(hose)【莫队算法裸题&&学习笔记】

    2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 9894  Solved: 4561[Subm ...

随机推荐

  1. notepad++ 查找引用(Find Reference)(适用于c c++及各类脚本比方lua、python等)

    在程序开发过程中,程序猿经经常使用到的一个功能就是查找引用(Find Reference).Visual Studio里面的相应功能是"查找全部引用"(Find All Refer ...

  2. MJRefresh实现刷新(使用它的Block方法)

    MJRefresh实现刷新(使用它的Block方法)   // // YFMVCPostListViewController.m // iOS122 // // Created by 颜风 on 15 ...

  3. Lnixu Bash

    一.简单命令 1.创建文件(vi) vi hellowold.txt2.创建目录(mkdir) mkdir linux_bash3.删除文件(rm) rm helloworld.txt4.复制文件(c ...

  4. 嵌入式开发之davinci--- 8148/8168/8127 中的图像缩放sclr、swms之后出现图像视频卡顿、屏幕跳跃的问题

    ()问题原因 这边的case链路是这样的camera->sclr(yuv420sp cif)->dup->ipcframeoutm3<->ipcframerocess&l ...

  5. 【bzoj4401】块的计数

    首先,块的大小确定的话,可以发现方案最多只有1种 然后就可以O(nsqrt(n))搞,不过会TLE 接着我们又发现,一个节点可以作一个块的根,当且仅当该节点的size能被块的大小整除 然后就可以O(n ...

  6. 3.myeclipse 8.5 m1 注册码

    为了能在eclipse 中方便的使用uml,尝试了多次安装各种uml插件未果,myeclipse 自带uml插件,但是要注册啊,要破解啊!!! user:baiduzhidaopassword:oLR ...

  7. Why Do Microservices Need an API Gateway?

    Why Do Microservices Need an API Gateway? - DZone Integration https://dzone.com/articles/why-do-micr ...

  8. Entity Framework底层操作封装V2版本号(3)

    如今是附加的,组合查询须要的扩展类.大家知道lanmda表达式的组合条件比較麻烦,所以就加了一样一个类,方便进行组合查询: using System; using System.Collections ...

  9. ubuntu切换中英文通用方法,ubuntu中文语言

    1:点击桌面右上角的齿轮,选择“system settings”进入系统设置界面

  10. 四.OC基础--1.文档安装和方法重载,2.self和super&static,3.继承和派生,4.实例变量修饰符 ,5.私有变量&私有方法,6.description方法

    四.OC基础--1.文档安装和方法重载, 1. 在线安装 xcode-> 系统偏好设置->DownLoads->Doucument->下载 2. 离线安装 百度xcode文档 ...