莫队算法学习笔记【BZOJ2038:小Z的袜子】【SPOJ3267:D-query】
很久以前傻乎乎地看来源奇怪的资料的时候被各种曼哈顿弄晕了。
然后现在学会的是分块方法。另新创一个分块方法。
让我们考虑这样一个区间询问问题……
它有如下的性质:
0,n个数,Q个询问。
1,它没有修改操作,这意味着我们可以按我们喜欢的次序跟询问玩耍。实际上后面会讲到我们完全可以按任意次序玩耍。
2,如果我们知道区间询问 [L , R] 对应的值,我们可以轻易求出 [L±1 , R] 和 [L , R±1] 的值。
(其实如果限制增加,比如只能求 [L+1 , R] 和 [L , R-1] 的值,同样不影响问题的解决,——这是我口胡的,没写过,日后再说。我不知道在传统莫队中是怎样的,在新算法中可行。)
3,若2中操作对应复杂度为P,而你需要用O(P*n^1.5)(明明是)O(P*n*Q^0.5)的复杂度来解决问题,那么莫队算法就是你需要的了。
离线方法:将n个数分成sqrt(n)块(一堆这么说的人什么心态)sqrt(Q)块
询问按区间排序,以L所在块序号为第一关键字,R为第二关键字,进行排序。(传统莫队就是如此搞法,YY一下就好了)
不过不知道这个做法怎么流传开的,感觉毫无优点吗……尤其是复杂度这么误导真的好吗。
大部分此类区间题Q总是大于N的(大很多!……)。然后正确的曼哈顿树求法最坏是n*sqrt(m),分块也可以做到这个复杂度,但无脑分块直接成了m*sqrt(n)……
总有一天要出摸你赛把你们卡光光……
(
但是话说回来……随机数据里sqrt(n)块好像还要快一点……
随机数据里明显是把块弄得越大越好吗……
)
先说【SPOJ3267:D-query】,毕竟是要做这道才会的。
不过能想用莫队A这题的也不多。
题目描述
输入
第一行:一个整数n,表示数列的长度。
第二行:n个整数a1,a2,a3,···,an(1 <= ai <= 10^6),数与数之间用一个空格隔开。
第三行:一个整数q(<= 200000),表示询问的次数。
接下来q行,每行两个整数i,j(1 <= i <= j <= n <= 30000),表示要询问的区间。
输出
对于每一询问,输出在这个区间内不同的数字个数。
样例输入
样例输出
无特别。
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstring>
using
namespace
std;
#define rep(i,j,n) for(int i=j;i<=n;i++)
char
c;
template
<
class
T>
inline
void
read(T&x){
for
(c=
getchar
();c<
'0'
||c>
'9'
;c=
getchar
());
for
(x=0;c>=
'0'
&&c<=
'9'
;c=
getchar
())x=x*10+c-
'0'
;};
struct
E{
int
x,y,l,list;}d[200010];
int
next[30010],prev[30010],p[1000010],ans[200010];
int
n,nn,x,T;
bool
cmp(E x,E y){
return
x.l<y.l||(x.l==y.l&&x.y<y.y);}
int
main(){
read(n);
memset
(next,63,n<<2);
rep(i,1,n){
read(x);
if
(!p[x])p[x]=i;
else
prev[i]=p[x],next[p[x]]=i,p[x]=i;
}
read(T);nn=
int
(
sqrt
(n));//块长……理论上的话……改成n/sqrt
(m)比较好
rep(i,1,T){read(d[i].x);read(d[i].y);d[i].l=(d[i].x+nn-1)/nn;d[i].list=i;}
sort(d+1,d+T+1,cmp);
int
ll,lr,now;
rep(ii,1,T){
if
(d[ii].l!=d[ii-1].l){
now=0;
rep(i,d[ii].x,d[ii].y)
if
(prev[i]<d[ii].x) now++;
ans[d[ii].list]=now;
}
else
{
if
(d[ii-1].x<d[ii].x){
ll=d[ii-1].x;lr=d[ii].x-1;
rep(i,ll,lr)
if
(next[i]>d[ii-1].y) now--;
}
else
{
ll=d[ii].x;lr=d[ii-1].x-1;
rep(i,ll,lr)
if
(next[i]>d[ii-1].y) now++;
}
ll=d[ii-1].y+1;lr=d[ii].y;
rep(i,ll,lr)
if
(prev[i]<d[ii].x) now++;
ans[d[ii].list]=now;
}
}
rep(i,1,T)
printf
(
"%d\n"
,ans[i]);
}
莫队算法学习笔记【BZOJ2038:小Z的袜子】【SPOJ3267:D-query】的更多相关文章
- 【序列莫队】BZOJ2038- [2009国家集训队]小Z的袜子(hose)
[题目大意]给出1-N只袜子的颜色,多次询问L-R中选出一双同色袜子的概率. [思路] 裸莫队.基本的莫队步骤:①分组(每组大小为根号sqrt(n),共sqrt(n)组)②排序(左边界分组,右边界在组 ...
- E. XOR and Favorite Number 莫队 2038: [2009国家集训队]小Z的袜子(hose)
一直都说学莫队,直到现在才学,训练的时候就跪了 T_T,其实挺简单的感觉.其实训练的时候也看懂了,一知半解,就想着先敲.(其实这样是不好的,应该弄懂再敲,以后要养成这个习惯) 前缀异或也很快想出来 ...
- BZOJ2038 小Z的袜子 (莫队算法)
题目链接: http://www.lydsy.com/JudgeOnline/problem.php?id=2038 专题练习: http://acm.hust.edu.cn/vjudge/conte ...
- BZOJ-2038 小Z的袜子(hose) 莫队算法
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MB Submit: 5573 Solved: 2568 [Subm ...
- [BZOJ2038]小Z的袜子(莫队算法)
题目:http://www.lydsy.com/JudgeOnline/problem.php?id=2038 分析:莫队算法 莫队算法是一种思想…… 处理问题:不带修改的区间询问 使用要求:[l-1 ...
- bzoj2038 小Z的袜子(hose)——莫队算法
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2038 就是莫队算法: 先写了个分块,惨WA: #include<iostream> ...
- 洛谷 P1972 [SDOI2009]HH的项链【莫队算法学习】
P1972 [SDOI2009]HH的项链 题目背景 无 题目描述 HH 有一串由各种漂亮的贝壳组成的项链.HH 相信不同的贝壳会带来好运,所以每次散步完后,他都会随意取出一段贝壳,思考它们所表达的含 ...
- BZOJ2038小Z的袜子(hose)
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 2343 Solved: 1077[Subm ...
- Bzoj2038 小Z的袜子(hose)
Time Limit: 20000MS Memory Limit: 265216KB 64bit IO Format: %lld & %llu Description 作为一个生活散漫 ...
随机推荐
- 淘宝接口实现ip归属地查询
<?php header('content-type:text/html;charset=utf-8'); /*获取当前ip归属地 调用淘宝接口 */ function get_ip_place ...
- 【结构型】Flyweight模式
享元模式的主要目的.意图是为对象的大量使用提供一种共享机制.该模式的思想重在复用.共享复用.像文字.列表中的格子等这类多数都是需要考虑复用技术,否则必将大量耗费内存空间而使资源以及性能等大量耗费.该模 ...
- d007: 求两数的整数商 和 余数
内容: 求两数的整数商 和 余数 输入说明: 一行两个整数 输出说明: 一行两个整数 输入样例: 18 4 输出样例 : 4 2 #include <stdio.h> int main ...
- 针对access数据库的增删改查
1.执行查询操作:(ExecuteReader方法) string myConnectionString = "Provider = Microsoft.Jet.OLEDB.4.0;Data ...
- mysql undo
mysql> show variables like '%undo%'; +-------------------------+-------+ | Variable_name | Value ...
- BZOJ3389: [Usaco2004 Dec]Cleaning Shifts安排值班
3389: [Usaco2004 Dec]Cleaning Shifts安排值班 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 45 Solved: ...
- Java中string拼接,StringBuilder,StringBuffer和+
Java中string拼接,StringBuilder,StringBuffer和+,到底哪个更合适? StringBuilder线程不安全,效率较线程安全的StringBuffer高.jdk1.5之 ...
- Qt入门(2)——使用Qt编写的Hello world
对于一个qt初学者来说,一步一步来直到作出一个hello world是最基础的入门. 从菜单:应用程序->编程中找到QtCreator
- ASCII码表(0 - 255)
目前计算机中用得最广泛的字符集及其编码,是由美国国家标准局(ANSI)制定的ASCII码(American Standard Code for Information Interchange,美国标准 ...
- socket实例2
第二个实例创建一个java工程,基于tomcat服务器,程序运行时会启动客户端,实现了一个客户端向其他的客户端发送即时信息的功能 MainWindow.java package com.jikexue ...