浅谈莫队

借用题目: 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. Leetcode题库——12.整数转罗马数字

    @author: ZZQ @software: PyCharm @file: intToRoman.py @time: 2018/9/28 21:59 要求: 字符 数值 I 1 V 5 X 10 L ...

  2. Java第二天——标识符命名规则、Java的知识、快捷键的使用、Scanner获取值的常用方法

    1.标识符命名规则 字母.下划线.数字.美元符号($)由这四个部分组成. 标识符=首字母+其他 首字母:字母.下划线.美元符号($) 其他:字母.下划线.数字.美元符号($) 注意: 1.首字母不能为 ...

  3. Scala入门系列(六):面向对象之object

    object object相当于class的单个实例,类似于Java中的static,通常在里面放一些静态的field和method.   第一次调用object中的方法时,会执行object的con ...

  4. angularJS1笔记-(18)-$http及用angular实现JSONP跨域访问过程

    官网上的解释为: The $http service is a core AngularJS service that facilitates communication with the remot ...

  5. Navicat for MySQL笔记1

    1.MySQL数据库的基本操作 A.系统数据库 安装MySQL数据库服务器后,自带的数据库. B.用户数据库 用户根据实际需求所创建的数据库. C.数据库对象 表.视图.存储过程.函数.触发器以及事件 ...

  6. TCP连接 三次握手 四次挥手

    前言: TCP协议是面向连接.安全可靠.基于字节流的传输层协议,在进行http协议访问时就用到了tcp连接.在建立TCP连接时需要经历三次握手,断开连接时需要经历四次挥手.在此进行记录. 内容: TC ...

  7. 词频统计的java实现方法——第一次改进

    需求概要 原需求 1.读取文件,文件内包可含英文字符,及常见标点,空格级换行符. 2.统计英文单词在本文件的出现次数 3.将统计结果排序 4.显示排序结果 新需求: 1.小文件输入. 为表明程序能跑 ...

  8. HDU 2078 复习时间

    http://acm.hdu.edu.cn/showproblem.php?pid=2078 Problem Description 为了能过个好年,xhd开始复习了,于是每天晚上背着书往教室跑.xh ...

  9. Caffe使用step by step:使用自己数据对已经训练好的模型进行finetuning

    在经过前面Caffe框架的搭建以及caffe基本框架的了解之后,接下来就要回到正题:使用caffe来进行模型的训练. 但如果对caffe并不是特别熟悉的话,从头开始训练一个模型会花费很多时间和精力,需 ...

  10. 我的虚拟机中的 centOS 连不了网了

    网上的办法试过了,查看虚拟机的网络配置,是 NET的, 也 cd 到/etc/sysconfig/network-script/ifcfg-eth0 里面看了,onboot 本来就是 yes,要不然我 ...