3809: Gty的二逼妹子序列
Time Limit: 80 Sec Memory Limit: 28 MB
Submit: 1387 Solved: 400
[Submit][Status][Discuss]
Description
Input
Output
对每个询问,单独输出一行,表示sl...sr中权值∈[a,b]的权值的种类数。
Sample Input
4 4 5 1 4 1 5 1 2 1
5 9 1 2
3 4 7 9
4 4 2 5
2 3 4 7
5 10 4 4
3 9 1 1
1 4 5 9
8 9 3 3
2 2 1 6
8 9 1 4
Sample Output
0
0
2
1
1
1
0
1
2
HINT
Source
初识莫队算法,大体记录一下。
莫队算法可以用来解决一类区间询问问题,例如一道经典的例题:
给出一个序列,还有若干次询问,每次询问在区间[l,r]中有多少个数字出现了3次及3次以上。
先考虑暴力算法,不难想到对于每个询问,扫描一遍区间,用数组记录下每个数字出现的次数并及时统计出现3次及以上的数字个数,时间复杂度O(询问数*区间大小)。
再考虑高级算法,就是维护一段区间内的数字出现次数以及3次及三次以上的数字个数,区间每次可以O(1)的向某个方向(左或右)扩展一个元素,或弹出一个元素,只需要修改该数字的出现次数,并检查是否发生了从3到2或从2到3的“质变”即可。这个算法相较于上一个暴力算法并没与在复杂度上体现出什么优势,但这是莫队算法的基础。
最后看莫队算法,采用类似于分块的sqrt(n)划分方式,先将所有询问离线,按照询问的区间左端点排序,每sqrt(n)个单位长度划分为一组,注意是按照长度。然后对于每一组询问,在组内对询问按照区间右端点排序,使其单调,然后暴力处理一个组内的所有询问即可。由于左端点相距至多sqrt(n)个长度,所有每次维护的区间的左端点最多进行sqrt(n)次改变(加入元素或删除元素),而区间的右端点由于单调性质至多移动n个单位长度,全局复杂度O(n*sqrt(n)),优秀之极。
对于这道题,一开始的想法是莫队算法+树状数组(或线段树)什么的,时间复杂度O(N*sqrt(N)*log(N)),然而亲身实践之后并没有卡过去,看来出题人没有那么友好。
考虑把树状数组的O(logN)加入和O(logN)查询做一些调整,用分块的O(1)插入和O(sqrt(N))查询替代,全局复杂度降至O(NsqrtN)。
#include <bits/stdc++.h> template <class T>
inline void read(T &num) {
register int neg = false;
register int bit = getchar(); while (bit <= '') {
if (bit == '-')
neg ^= neg;
bit = getchar();
} num = ; while (bit >= '') {
num = num*
+ bit - '';
bit = getchar();
} if (neg)num = -num;
} const int N = 1e5 + ;
const int M = 1e6 + ; int n, m, s, num[N], cnt[N], sgl[N], sum[N]; /*<--- QRY --->*/ struct query {
int l, r, a, b, id, ans;
}qry[M]; inline bool cmp_lr(const query &A, const query &B) {
if (A.l / s != B.l / s)
return A.l < B.l;
else
return A.r < B.r;
} inline bool cmp_id(const query &A, const query &B) {
return A.id < B.id;
} /*<--- MO --->*/ inline int ask(int a, int b) {
if (a / s == b / s) {
int ret = ;
for (int i = a; i <= b; ++i)ret += sgl[i];
return ret;
}
else {
int ret = , lt = a / s + , rt = b / s - ;
for (int i = lt; i <= rt; ++i)ret += sum[i];
for (int i = a; i / s < lt; ++i)ret += sgl[i];
for (int i = b; i / s > rt; --i)ret += sgl[i];
return ret;
}
} inline void add(int k, int v) {
sgl[k] += v, sum[k/s] += v;
} inline void insert(int k) {
if (++cnt[k] == )add(k, );
} inline void remove(int k) {
if (--cnt[k] == )add(k, -);
} /*<--- MAIN --->*/ signed main(void) {
read(n);
read(m); s = sqrt(n); for (int i = ; i <= n; ++i)
read(num[i]); for (int i = ; i <= m; ++i) {
qry[i].id = i;
read(qry[i].l);
read(qry[i].r);
read(qry[i].a);
read(qry[i].b);
} memset(cnt, , sizeof(cnt));
memset(sum, , sizeof(sum));
memset(sgl, , sizeof(sgl)); std::sort(qry + , qry + + m, cmp_lr); for (int i = , x = , y = ; i <= m; ++i) {
while (x < qry[i].l)remove(num[x]), ++x;
while (y > qry[i].r)remove(num[y]), --y;
while (x > qry[i].l)--x, insert(num[x]);
while (y < qry[i].r)++y, insert(num[y]);
qry[i].ans = ask(qry[i].a, qry[i].b);
} std::sort(qry + , qry + + m, cmp_id); for (int i = ; i <= m; ++i)
printf("%d\n", qry[i].ans);
}
@Author: YouSiki