浅谈莫队

借用题目: bzoj 2038 2009 国家集训队 小Z的袜子http://www.lydsy.com/JudgeOnline/problem.php?id=2038
[2009国家集训队]小Z的袜子(hose)

http://www.lydsy.com/JudgeOnline/problem.php?id=2038

Time Limit:  Sec  Memory Limit:  MB

Description

作为一个生活散漫的人,小Z每天早上都要耗费很久从一堆五颜六色的袜子中找出一双来穿。终于有一天,小Z再也无法忍受这恼人的找袜子过程,于是他决定听天由命……
具体来说,小Z把这N只袜子从1到N编号,然后从编号L到R(L 尽管小Z并不在意两只袜子是不是完整的一双,甚至不在意两只袜子是否一左一右,他却很在意袜子的颜色,毕竟穿两只不同色的袜子会很尴尬。
你的任务便是告诉小Z,他有多大的概率抽到两只颜色相同的袜子。当然,小Z希望这个概率尽量高,所以他可能会询问多个(L,R)以方便自己选择。 Input 输入文件第一行包含两个正整数N和M。N为袜子的数量,M为小Z所提的询问的数量。接下来一行包含N个正整数Ci,其中Ci表示第i只袜子的颜色,相同的颜色用相同的数字表示。再接下来M行,每行两个正整数L,R表示一个询问。 Output 包含M行,对于每个询问在一行中输出分数A/B表示从该询问的区间[L,R]中随机抽出两只袜子颜色相同的概率。若该概率为0则输出0/,否则输出的A/B必须为最简分数。(详见样例) Sample Input Sample Output /
/
/
/
【样例解释】
询问1:共C(,)=10种可能,其中抽出两个2有1种可能,抽出两个3有3种可能,概率为(+)/=/=/。
询问2:共C(,)=3种可能,无法抽到颜色相同的袜子,概率为0/=/。
询问3:共C(,)=3种可能,均为抽出两个3,概率为3/=/。
注:上述C(a, b)表示组合数,组合数C(a, b)等价于在a个不同的物品中选取b个的选取方案数。
【数据规模和约定】
%的数据中 N,M ≤ ;
%的数据中 N,M ≤ ;
%的数据中 N,M ≤ , ≤ L < R ≤ N,Ci ≤ N。

题目简介

莫队算法
 
本方法由莫涛提出,故尊称为莫队算法
 
使用前提:
1、离线操作
2、若已知[l,r]内的答案,可以在O(1)时间内得到[l-1,r],[l+1,r],[l,r-1],[l,r+1]的答案,即可使用莫队算法,时间复杂度O(n^1.5)。
   若可在O(logn)内移动区间,时间复杂度则为O(n^1.5*logn)
 
个人理解:
莫队算法本质是分块
通过左右指针的移动避免了重复状态的计算
 
就本题来说
sum表示区间内数i的出现次数
那么∑sum[i]=r-l+1 
           2*(r-l+1)!
分母=  -----------   = (r-l)*(r-l+1)
         2!*(r-l-1) !
可以O(1)解决
那么就差∑sum[i]²
本题满足离线操作,如果我们能证明∑sum[i]²可以O(1)或O(logn)求出来,就可以使用莫队算法
由此图可以看出,满足前提条件2
所以我们可以使用莫队算法
还有,如果先询问[1,n],再问[1,1],再问[n,n],时间复杂度不还是n²吗
所以,为了避免指针在整个序列中移动,我们需要分块
 
因为分块保证了莫队算法的时间复杂度
 
如何分块?
令块的大小为根号n
先根据询问区间左端点所在块,从小到大排,这样就将整个操作序列划分为了根号n块
对于每个块内的询问,根据右端点从小到大排
然后我们从左到右顺序计算每个询问的答案
这样可以保证时间复杂度为O(n^1.5)
 
时间复杂度分析:
A.左端点
  1.块内的,由于块的大小为根号n,所以同一块内左端点一次询问最多移动根号n,即n^0.5次,m次询问总移动O(n*n^0.5)=O(n^1.5)
  2、不在同一块内的,以为块的大小为n^0.5,所以跨越一次最多加上O(n^0.5),最多能跨越(n^0.5)次,所以跨越块导致左端点最多加上O(n)
 所以左端点总的移动次数为O(n^1.5)+O(n)
B.右端点
 1、块内的,右端点最多从1移动到n,有根号n个块,所以右端点最多移动O(n^1.5)次
 2、跨越块的,最多跨越个根号n次,每次跨越最多移动n次,所以右端点最多移动 O(n^1.5)
所以右端点总移动次数为O(n^1.5)+O(n^1.5)
所以时间复杂度为O(n^1.5)
#include<cstdio>
#include<algorithm>
#include<cmath>
using namespace std;
int col[];
struct node
{
int l,r,id,pos;
long long a,b;
}e[];
int n,m,S;
long long h,ans,sum[];
bool cmp(node p,node q)
{
if(p.pos!=q.pos) return p.pos<q.pos;
return p.r<q.r;
}
bool id(node p,node q)
{
return p.id<q.id;
}
void update(int x,int k)
{
ans-=sum[col[x]]*sum[col[x]];
sum[col[x]]+=k;
ans+=sum[col[x]]*sum[col[x]];
}
long long gcd(long long a,long long b)
{
return b== ? a:gcd(b,a%b);
}
void solve()
{
int l=,r=;
for(int i=;i<=m;i++)
{
while(l<e[i].l) update(l++,-);
while(l>e[i].l) update(--l,);
while(r<e[i].r) update(++r,);
while(r>e[i].r) update(r--,-);
e[i].a=ans-(r-l+);
e[i].b=1ll*(r-l)*(r-l+);
h=gcd(e[i].a,e[i].b);
e[i].a/=h;e[i].b/=h;
}
}
int main()
{
scanf("%d%d",&n,&m);
S=sqrt(n);
for(int i=;i<=n;i++) scanf("%d",&col[i]);
for(int i=;i<=m;i++)
{
scanf("%d%d",&e[i].l,&e[i].r);
e[i].id=i;
e[i].pos=(e[i].l-)/S+;
}
sort(e+,e+m+,cmp);
solve();
sort(e+,e+m+,id);
for(int i=;i<=m;i++) printf("%lld/%lld\n",e[i].a,e[i].b);
}

小细节:l最初从1开始,r从0开始

因为l最开始一定小于e[i].l,这种情况下先计算l,在自加

r最开始一定小于e[i].r,这种情况下先自加,再计算

[2009国家集训队]小Z的袜子(hose) 浅谈莫队的更多相关文章

  1. [BZOJ2038]:[2009国家集训队]小Z的袜子(hose)(离线莫队)

    题目传送门 题目描述 作为一个生活散漫的人,小$Z$每天早上都要耗费很久从一堆五颜六色的袜子中找出一双来穿.终于有一天,小Z再也无法忍受这恼人的找袜子过程,于是他决定听天由命……具体来说,小$Z$把这 ...

  2. [2009国家集训队]小Z的袜子(hose)(BZOJ2038+莫队入门题)

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=2038 题目: 题意:中文题意,大家都懂. 思路:莫队入门题.不过由于要去概率,所以我们假 ...

  3. 【bzoj 2038】 [2009国家集训队]小Z的袜子(算法效率--莫队分块算法 模版题)

    题意:小Z有N只袜子,有不同的颜色.他有M个提问,问从编号为[L,R]的袜子中随机选一双同色的袜子的概率,用最简分数表示. 解法:经典的莫队算法--无修改.不强制在线(可离线).状态转移可以一步完成. ...

  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. 莫队算法 2038: [2009国家集训队]小Z的袜子(hose)

    链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2038 2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 ...

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

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

  8. BZOJ2038: [2009国家集训队]小Z的袜子(hose) -- 莫队算法 ,,分块

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

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

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

随机推荐

  1. arcgis for android apk太大

    原来大概都要20多M, 太大的原来是.so文件 arcgis for android api里面有armeabi armeabi-v7a  x86的 每个so都接近10m 要是都保留就20多m了 由于 ...

  2. python learning Exception & Debug.py

    ''' 在程序运行的过程中,如果发生了错误,可以事先约定返回一个错误代码,这样,就可以知道是否有错,以及出错的原因.在操作系统提供的调用中,返回错误码非常常见.比如打开文件的函数open(),成功时返 ...

  3. TensorFlow问题“Attempting to use uninitialized value”

    1.出现的问题: 对已经保存好的模型,在进行重载并继续训练的过程中出现了以下问题: 2.解决办法: 在查找了相关资料后,了解到,该错误是指在从tfrecord中读取数据时一些参数未被初始化,如果直接r ...

  4. PHP面试题一

    http://www.viphper.com/?p=28 1.用PHP打印出前一天的时间格式是2006-5-10 22:21:21(2分) $a = date("Y-m-d H:i:s&qu ...

  5. scp命令与Screen服务的区别

    scp:远程传输命令.(通过网络传送给其他主机,又恰好两台主机都是linux系统,便可以使用scp传输文件) 参数 作痛 -v 先是详细的连接进度 -P 指定远程主机的sshd端口号 -r 传送文件夹 ...

  6. 微信小程序wx:for和wx:for-item的正确用法

    wx:for="{{list}}"用来循环数组,而list即为数组名wx:for-item="items" 即用来定义一个循环过程中每个元素的变量的 如果是一维 ...

  7. 用vue实现省市县三级联动

    我真的没想到这个会困扰到我.最开始以为,不就是直接找个简单的插件就实现了吗,jquery插件找了几个,都没有达到目的. 需求是这样的: 点击input框,弹出一个popup,然后可以滚动选择省,市,县 ...

  8. PowerCLI

    最近需要用命令行操作VMWare,现将一些经常用的命令记录一下.安装VMWare命令很简单,不再像原来需要单独下载PowerCLI安装包,直接在Powershell Gallery里在线安装即可. # ...

  9. 【Linux笔记】CentOS 7 systemctl、firewalld

    一.CentOS7 systemctl 在CentOS7中,进行chkconfig命令操作时会发现有类似“systemctl.....”的提示,systemctl可以简单实现service和chkco ...

  10. UVALive3713_Astronauts

    有n个宇航员,根据年龄限制,所有宇航员只能从事A或B中的一种任务,所有人都可以从事C的任务.有的宇航员之间相互讨厌,不能分在一组,求出一种满足条件的分配方案. 2sat.mark[]中i+i和i+i+ ...