[知识点]数列分块入门1-9

一、前言

  对线段树还挺熟悉的我之前却从没写过分块的题(?)。然后兴致一来就决定搜点练习,找到了来自hzwer的《数列分块入门1-9》,觉得挺不错的,于是决定做做。

 

二、概念 / 作用

  概念:将数列等分为若干个不相交的区间,每一个区间称为一个块。

  作用:优化算法,降低复杂度。具体如何降低,在下面的题目中会逐步提及。题目呈难度递增趋势。

 

三、题目 / 代码

1、分块入门1(传送门:https://loj.ac/problem/6277)

  题面:给出一个长为 n 的数列,以及 n 个操作,操作涉及区间加法,单点查询。

  挺多数据结构均能实现的经典题目,譬如线段树。这里我们用分块来做。将 n 个元素等分为若干块,比如{1, 4, 8, 2, 9, 6, 3, 7, 5},等分为3块,则第一块包含的数据为{1, 4, 8},第二、三块以此类推。我们给每一个块增加一个加法标记,对于每次的区间[l, r]加法操作,直接对块进行标记叠加。

  l, r必然不一定是块的边界,也就意味着左右端点可能在块的中间,直接一个个暴力增加。设块的元素个数为m,标记块个数至多n / m个,暴力增加元素个数至多2m个,复杂度分析:O(n / m) + O(m),根据均值不等式,可证明m = √n时存在最低复杂度。故我们以下所有分块大小均默认为√n。

  询问就很轻松了,直接返回元素的值加上所在区间的标记。

 

代码:

 1 #include 
 2 using namespace std;
 3 
 4 #define MAXN 50005
 5 
 6 int n, a[MAXN], x, b[MAXN], f[MAXN];
 7 
 8 void add(int l, int r, int w) {
 9     for (int i = l; i <= min(b[l] * x, r); i++) a[i] += w;
10     if (b[l] != b[r])
11         for (int i = (b[r] - 1) * x + 1; i <= r; i++) a[i] += w;
12     for (int i = b[l] + 1; i <= b[r] - 1; i++) f[i] += w;
13 }
14 
15 int main() {
16     int o, l, r, w;
17     scanf("%d", &n), x = sqrt(n);
18     for (int i = 1; i <= n; i++) scanf("%d", &a[i]);
19     for (int i = 1; i <= n; i++) b[i] = (i - 1) / x + 1;
20     for (int i = 1; i <= n; i++) {
21         scanf("%d %d %d %d", &o, &l, &r, &w);
22         if (o) printf("%d\n", a[r] + f[b[r]]);
23         else add(l, r, w);
24     }
25     return 0;
26 } 

 

2、分块入门2(传送门:https://loj.ac/problem/6278)

  题面:给出一个长为 n 的数列,以及 n 个操作,操作涉及区间加法,询问区间内小于某个值 x 的元素个数。

  // 对于题目本身的分析就不多赘述了,因为hzwer已经分析的太好了,不在关公们面前耍刀了,所以需要具体的做题思路可转至hzwer的博客(见上)。

  hzwer原代码使用了vector,正好去了解了下vector(https://www.cnblogs.com/jinkun113/p/10691919.html)。当然此题不用vector问题也不大,这里把两个版本都贴出来了。

  不论是用普通数组还是vector,有一个需要注意的点是,对于每次的区间修改,左右端点所在的块会因修改而不再呈升序排列,所以需要重新维护。

 

代码 - 普通版:

 1 #include 
 2 using namespace std;
 3 
 4 #define MAXN 50005
 5 
 6 int n, x;
 7 int a[MAXN], b[MAXN], f[MAXN], c[505][505];
 8 
 9 void reset(int o) {
10     memset(c[o], 0, sizeof(c[o]));
11     for (int i = (o - 1) * x + 1; i <= min(o * x, n); i++) 
12         c[o][++c[o][0]] = a[i];
13     sort(c[o] + 1, c[o] + c[o][0] + 1);
14 }
15 
16 void add(int l, int r, int w) {
17     for (int i = l; i <= min(b[l] * x, r); i++) a[i] += w;
18     reset(b[l]);
19     if (b[l] != b[r]) {
20         for (int i = (b[r] - 1) * x + 1; i <= r; i++) a[i] += w;
21         reset(b[r]);
22     }
23     for (int i = b[l] + 1; i <= b[r] - 1; i++) f[i] += w;
24 }
25 
26 int query(int l, int r, int w) {
27     int ans = 0;
28     for (int i = l; i <= min(b[l] * x, r); i++)
29         if (a[i] + f[b[l]] < w) ans++;
30     if (b[l] != b[r])
31         for (int i = (b[r] - 1) * x + 1; i <= r; i++)
32             if (a[i] + f[b[r]] < w) ans++;
33     for (int i = b[l] + 1; i <= b[r] - 1; i++) {
34         int x = w - f[i];
35         for (int j = 1; j <= c[i][0]; j++)
36             if (c[i][j] < x) ans++;
37             else break;
38     }
39     return ans;
40 }
41 
42 int main() {
43     int o, l, r, w;
44     scanf("%d", &n), x = sqrt(n);
45     for (int i = 1; i <= n; i++) scanf("%d", &a[i]);
46     for (int i = 1; i <= n; i++)
47         b[i] = (i - 1) / x + 1, c[b[i]][++c[b[i]][0]] = a[i];
48     for (int i = 1; i <= b[n]; i++) sort(c[i] + 1, c[i] + c[i][0] + 1);
49     for (int i = 1; i <= n; i++) {
50         scanf("%d %d %d %d", &o, &l, &r, &w);
51         if (!o) add(l, r, w);
52         else printf("%d\n", query(l, r, w * w));
53     }
54     return 0;
55 }

 

代码 - vector版:

 1 #include 
 2 using namespace std;
 3 
 4 #define MAXN 50005
 5 
 6 int n, x;
 7 int a[MAXN], b[MAXN], f[MAXN];
 8 
 9 vector <int> v[505];
10 
11 void reset(int o) {
12     v[o].clear();
13     for (int i = (o - 1) * x + 1; i <= min(o * x, n); i++) v[o].push_back(a[i]);
14     sort(v[o].begin(), v[o].end());
15 }
16 
17 void add(int l, int r, int w) {
18     for (int i = l; i <= min(b[l] * x, r); i++) a[i] += w;
19     reset(b[l]);
20     if (b[l] != b[r]) {
21         for (int i = (b[r] - 1) * x + 1; i <= r; i++) a[i] += w;
22         reset(b[r]);
23     }
24     for (int i = b[l] + 1; i <= b[r] - 1; i++) f[i] += w;
25 }
26 
27 int query(int l, int r, int w) {
28     int ans = 0;
29     for (int i = l; i <= min(b[l] * x, r); i++)
30         if (a[i] + f[b[l]] < w) ans++;
31     if (b[l] != b[r])
32         for (int i = (b[r] - 1) * x + 1; i <= r; i++)
33             if (a[i] + f[b[r]] < w) ans++;
34     for (int i = b[l] + 1; i <= b[r] - 1; i++) {
35         int x = w - f[i];
36         ans += lower_bound(v[i].begin(), v[i].end(), x) - v[i].begin();
37     }
38     return ans;
39 }
40 
41 int main() {
42     int o, l, r, w;
43     scanf("%d", &n), x = sqrt(n);
44     for (int i = 1; i <= n; i++) scanf("%d", &a[i]);
45     for (int i = 1; i <= n; i++)
46         b[i] = (i - 1) / x + 1, v[b[i]].push_back(a[i]);
47     for (int i = 1; i <= b[n]; i++) sort(v[i].begin(), v[i].end());
48     for (int i = 1; i <= n; i++) {
49         scanf("%d %d %d %d", &o, &l, &r, &w);
50         if (!o) add(l, r, w);
51         else printf("%d\n", query(l, r, w * w));
52     }
53     return 0;
54 }

 

To be continued...

你可能感兴趣的:([知识点]数列分块入门1-9)