首页 > 其他分享 >分块莫队学习笔记

分块莫队学习笔记

时间:2025-01-18 14:45:49浏览次数:1  
标签:right 分块 int 复杂度 笔记 sqrt mathcal 莫队 left

优雅的暴力。

引入

link

这道题显然可以用线段树、树状数组做,但如果我偏不用这些数据结构呢?

我们知道,暴力修改和查询最坏是 \(\mathcal{O}(n)\) 的,这样肯定会挂掉。

那该怎么办呢?

正题

分块

考虑将序列分成若干块,我们设每块长为 \(B\)。

对于每次查询 \(\left [ l, r \right ]\),我们涉及到修改的块是 \(\left [ b_l, b_r \right ]\)(\(b_i\) 代表 \(i\) 属于哪个块)。

其中 \(\left [ b_l + 1, b_r - 1 \right ]\) 是整块都被修改了。

不妨设置一个懒标记,把每块的整块操作都加到这里面。

这样修改的复杂度是 \(\mathcal{O}(\frac{n}{B})\) 的。

那剩下的我们就可以暴力操作,复杂度是 \(\mathcal{O}(B)\) 的。

查询同理。

此时修改查询的复杂度就变成了 \(\mathcal{O}(B + \frac{n}{B})\) 了。

使得该数最小的显然是 \(B = \sqrt{n}\),所以该算法的时间复杂度是 \(\mathcal{O}(m\sqrt{n})\)。

分块主要解决区修区查类问题,只要满足以下条件即可:

  • 可以打懒标记(结合律)。
  • 时间复杂度允许。

优势:可解决问题范围广。

劣势:时间复杂度高。

时间复杂度:\(\mathcal{O}(m\sqrt{n})\)。

空间复杂度:\(\mathcal{O}(n)\)。

莫队

普通莫队

莫队是一种离线算法,需要满足以下条件:

  • 在知道 \(\left [ l, r \right ]\) 的答案的情况下,可以 \(\mathcal{O}(1)\) 求出 \(\left [ l, r + 1 \right ]\)、 \(\left [ l, r - 1 \right ]\)、 \(\left [ l + 1, r \right ]\)、 \(\left [ l - 1, r \right ]\) 的答案。
  • 允许离线。
  • 只有询问没有修改。

首先将所有的询问离线下来,记为 \(\left [ ql_1, qr_1 \right ],\left [ ql_2, qr_2 \right ],\dots,\left [ ql_m, qr_m \right ]\)。

将询问排序(这正是莫队算法的精髓),从上一个询问的答案一个个改到当前询问,得到答案。

实现:

for (int i = 1; i <= m; i++) {
	while (l < q[i].l) del(l++);
	while (r > q[i].r) del(r--);
	while (l > q[i].l) add(--l);
	while (r < q[i].r) add(++r);
	ans[q[i].id] = res;
}

但是仔细分析发现时间复杂度仍然可以被卡成 \(nm\),一点都不优秀,甚至会更慢。

考虑优化

我们想要优化复杂度的根本是让 \(l\) 和 \(r\) 指针移动的距离尽量少。

对询问范围进行分块,块长为 \(B\)。

以询问左端点的块编号为第一关键字,右端点为第二关键字排序。

  • 如果当前询问与上一次处于同一块,则 \(l\) 最多移动 \(B\)。
  • 不同块的询问,\(l\) 最多移动 \(2B\)。

则:

  • \(l\) 移动的复杂度是 \(m\times B = mB\);
  • \(r\) 的复杂度是 \(\frac{n}{B} \times n = \frac{n^2}{B}\)。

则复杂度是 \(\mathcal{O}(mB + \frac{n^2}{B})\)。

使得该式最小的 \(B\) 的值是 \(\frac{n}{\sqrt m}\),则此时的时间复杂度就是 \(\mathcal{O}(n\sqrt{m} + m\log m)\)。

\(m \log m\) 是排序的复杂度。

总结一下。

普通莫队解决的问题满足以下条件:

  • 在知道 \(\left [ l, r \right ]\) 的答案的情况下,可以 \(\mathcal{O}(1)\) 求出 \(\left [ l, r + 1 \right ]\)、 \(\left [ l, r - 1 \right ]\)、 \(\left [ l + 1, r \right ]\)、 \(\left [ l - 1, r \right ]\) 的答案。
  • 允许离线。
  • 只有询问没有修改。

优势:再没有更快的思维做法之前,她几乎是跑得最快并且思维含量最低的。

劣势:只支持离线。

时间复杂度: \(\mathcal{O}(n\sqrt{m} + m\log m)\)。

空间复杂度: \(\mathcal{O}(n)\)。

例题 1:小 B 的询问

非常板子的一道,维护一下 \(c\) 数组即可。

#include <bits/stdc++.h>
// #define int long long
#define pii pair<int, int>
#define FRE(x) freopen(x ".in", "r", stdin), freopen(x ".out", "w", stdout)
#define ALL(x) x.begin(), x.end()
using namespace std;

int _test_ = 1;

const int N = 50008;

int n, m, k, block_size, res, cnt[N], a[N], ans[N];
struct node {
	int l, r, id;
} q[N];

bool operator<(node x, node y) {
	int xl = (x.l - 1) / block_size + 1, xr = (x.r - 1) / block_size + 1;
	int yl = (y.l - 1) / block_size + 1, yr = (y.r - 1) / block_size + 1;
	return (xl != yl) ? (xl < yl) : (x.r < y.r);
}

void add(int x) {
	res += cnt[a[x]] * 2 + 1;
	cnt[a[x]]++;
}

void del(int x) {
	res -= cnt[a[x]] * 2 - 1;
	cnt[a[x]]--;
}

void init() {}

void clear() {}

void solve() {
	cin >> n >> m >> k;
	for (int i = 1; i <= n; i++) {
		cin >> a[i];
	}
	block_size = n / sqrt(m); // 块长
	for (int i = 1; i <= m; i++) {
		cin >> q[i].l >> q[i].r;
		q[i].id = i;
	}
	sort(q + 1, q + m + 1);
	int l = 1, r = 0;
	for (int i = 1; i <= m; i++) {
		while (l < q[i].l) del(l++);
		while (r > q[i].r) del(r--);
		while (l > q[i].l) add(--l);
		while (r < q[i].r) add(++r);
		ans[q[i].id] = res;
	}
	for (int i = 1; i <= m; i++) {
		cout << ans[i] << "\n";
	}
}

signed main() {
	ios::sync_with_stdio(0);
	cin.tie(0), cout.tie(0);
//	cin >> _test_;
	init();
	while (_test_--) {
		clear();
		solve();
	}
	return 0;
}

不过此题块长就是 \(1\) 都能在 \(700\) 毫秒以内过,数据太水。

例题 2:小 Z 的袜子

也是非常板子的一道,维护一下 \(c\) 数组,并将上一题中的答案分别记分子分母即可。

请注意分子为 \(0\) 的情况。

#include <bits/stdc++.h>
// #define int long long
#define pii pair<int, int>
#define FRE(x) freopen(x ".in", "r", stdin), freopen(x ".out", "w", stdout)
#define ALL(x) x.begin(), x.end()
using namespace std;

int _test_ = 1;

const int N = 500008;

int n, m, k, block_size, len;
pii res;
int cnt[N], a[N];
pii ans[N];
struct node {
	int l, r, id;
} q[N];

bool operator<(node x, node y) {
	int xl = (x.l - 1) / block_size + 1, xr = (x.r - 1) / block_size + 1;
	int yl = (y.l - 1) / block_size + 1, yr = (y.r - 1) / block_size + 1;
	return (xl != yl) ? (xl < yl) : (x.r < y.r);
}

void add(int x) {
	res.first += cnt[a[x]];
	res.second += len;
	len++;
	cnt[a[x]]++;
}

void del(int x) {
	len--;
	cnt[a[x]]--;
	res.first -= cnt[a[x]];
	res.second -= len;
}

void init() {}

void clear() {}

void solve() {
	cin >> n >> m;
	for (int i = 1; i <= n; i++) {
		cin >> a[i];
	}
	block_size = n / sqrt(m);
	for (int i = 1; i <= m; i++) {
		cin >> q[i].l >> q[i].r;
		q[i].id = i;
	}
	sort(q + 1, q + m + 1);
	int l = 1, r = 0;
	for (int i = 1; i <= m; i++) {
		if (q[i].l == q[i].r) ans[q[i].id] = {0, 1};
		while (l < q[i].l) del(l++);
		while (r > q[i].r) del(r--);
		while (l > q[i].l) add(--l);
		while (r < q[i].r) add(++r);
		if (res.first == 0) {
			ans[q[i].id] = {0, 1};
			continue;
		}
		int g = __gcd(res.first, res.second);
		ans[q[i].id] = {res.first / g, res.second / g};
	}
	for (int i = 1; i <= m; i++) {
		cout << ans[i].first << "/" << ans[i].second << "\n";
	}
}

signed main() {
	ios::sync_with_stdio(0);
	cin.tie(0), cout.tie(0);
//	cin >> _test_;
	init();
	while (_test_--) {
		clear();
		solve();
	}
	return 0;
}

事实证明,还是 \(B = \frac{n}{\sqrt{m}}\) 跑得最快。

带修莫队

由于不能带修改实在是太别扭了,所以出现了带修莫队

带修莫队的思想跟所有可持久化数据结构是差不多的。

link.

由于加进了修改,我们无法再像正常莫队一样转移了。

可以考虑在迭代时增加一维时间戳。

每次就按顺序一个一个增加或减少修改即可。

同时就要以右端点所在块编号为第二关键字、时间为第三关键字排序。

时间复杂度与最优块长

设块长为 \(B\)、序列长度为 \(n\)、询问次数为 \(q\)、修改次数为 \(c\)。

  • 左右端点移动上文分析过,是 \(qB + \frac{n^2}{B}\) 的。
  • 时间指针,对于每一个块,我们至多移动 \(c\) 次,即 \(\frac{n}{B} \times \frac{n}{B} \times c = \frac{cn^2}{B^2}\)。

总时间复杂度为 \(\mathcal{O}(qB + \frac{n^2}{B} + \frac{cn^2}{B^2})\)。

最优块长大概是……

\[\frac{n^2}{3^{1/3}(9m^3n^2+\sqrt{3}\sqrt{27m^6n^4-m^3n^6})^{1/3}}+\frac{(9m^3n^2+\sqrt{3}\sqrt{27m^6n^4-m^3n^6})^{1/3}}{3^{2/3}m} \]

所以还是取一个更好看一点的。

譬如 \(B = \sqrt[3]{n^2}\)。

所以此时时间复杂度是约 \(\mathcal{O}(\sqrt[3]{n^5})\)。


总结一下,带修莫队需要满足以下条件:

  • 在知道 \(\left [ l, r \right ]\) 的答案的情况下,可以 \(\mathcal{O}(1)\) 求出 \(\left [ l, r + 1 \right ]\)、 \(\left [ l, r - 1 \right ]\)、 \(\left [ l + 1, r \right ]\)、 \(\left [ l - 1, r \right ]\) 的答案。
  • 允许离线。

优势:可以允许修改。

劣势:比思维方法慢且只能离线。、

时间复杂度:\(\mathcal{O}(n\log n + \sqrt[3]{n^5})\)​。

空间复杂度: \(\mathcal{O}(n)\)。

例题1:数颜色 / 维护队列

按上文中写的模拟即可。

#include <bits/stdc++.h>
#define int long long
#define pii pair<int, int>
#define FRE(x) freopen(x ".in", "r", stdin), freopen(x ".out", "w", stdout)
#define ALL(x) x.begin(), x.end()
using namespace std;

int _test_ = 1;

const int N = 2e6 + 5; 

int n, m, block_size, cnt_c, cnt_q, a[N], bel[N], cnt[N], ans[N], res;
struct query {
	int l, r, t, id;
} c[N], q[N];
bool operator<(query x, query y) {
	return (bel[x.l] != bel[y.l]) ? (x.l < y.l) : ((bel[x.r] != bel[y.r]) ? (x.r < y.r) : (x.t < y.t));
}
void build() {
	block_size = pow(n, 0.666);
	for (int i = 1; i <= n; i++) {
		bel[i] = (i - 1) / block_size + 1;
	}
}
void add(int x) {
	res += (cnt[x] == 0);
	cnt[x]++;
}
void del(int x) {
	cnt[x]--;
	res -= (cnt[x] == 0);
}
void upt(int x, int y) {
	if (q[y].l <= c[x].l && c[x].l <= q[y].r) {
		del(a[c[x].l]);
		add(c[x].r);
	}
	swap(a[c[x].l], c[x].r);
}

void init() {}

void clear() {}

void solve() {
	cin >> n >> m;
	for (int i = 1; i <= n; i++) {
		cin >> a[i];
	}
	build();
	for (int i = 1; i <= m; i++) {
		char op;
		int l, r;
		cin >> op >> l >> r;
		if (op == 'Q') q[++cnt_q] = {l, r, cnt_c, cnt_q};
		else c[++cnt_c] = {l, r, 0, 0};
	}
	sort(q + 1, q + cnt_q + 1);
	int l = 1, r = 0, t = 0;
	for (int i = 1; i <= cnt_q; i++) {
		while (l > q[i].l) add(a[--l]);
		while (r < q[i].r) add(a[++r]);
		while (l < q[i].l) del(a[l++]);
		while (r > q[i].r) del(a[r--]);
		while (t < q[i].t) upt(++t, i);
		while (t > q[i].t) upt(t--, i); 
		ans[q[i].id] = res;
	}
	for (int i = 1; i <= cnt_q; i++) cout << ans[i] << "\n";
}

signed main() {
  ios::sync_with_stdio(0);
  cin.tie(0), cout.tie(0);
  // cin >> _test_;
  init();
  while (_test_--) {
    clear();
    solve();
	}
  return 0;
}

标签:right,分块,int,复杂度,笔记,sqrt,mathcal,莫队,left
From: https://www.cnblogs.com/zphh/p/18553906

相关文章

  • Vue3+TS笔记
    创建工程:npminitvue@latestVue3工程结构在main.js中:引入的vue更轻量,引入vue是一个更精简版的名为createApp的工厂函数import{createApp}from'vue'importAppfrom'./App.vue'createApp(App).mount('#app')vm实例对象上有一个mount方法,不是原型上的$mou......
  • 【STM32-学习笔记-7-】USART串口通信
    文章目录USART串口通信Ⅰ、硬件电路Ⅱ、常见的电平标准Ⅲ、串口参数及时序Ⅳ、STM32的USART简介数据帧起始位侦测数据采样波特率发生器Ⅴ、USART函数介绍Ⅵ、USART_InitTypeDef结构体参数1、USART_BaudRate2、USART_WordLength3、USART_StopBits4、USART_Parity5、USART......
  • 【STM32-学习笔记-8-】I2C通信
    文章目录I2C通信Ⅰ、硬件电路Ⅱ、IIC时序基本单元①起始条件②终止条件③发送一个字节④接收一个字节⑤发送应答⑥接收应答Ⅲ、IIC时序①指定地址写②当前地址读③指定地址读Ⅳ、MPU6050---6轴姿态传感器(软件I2C)1、模块内部电路2、寄存器地址3、软件模拟IIC①......
  • 【STM32-学习笔记-9-】SPI通信
    文章目录SPI通信Ⅰ、SPI通信概述1、SPI技术规格2、SPI应用3、硬件电路移位示意图Ⅱ、SPI时序基本单元①、起始条件②、终止条件③、交换一个字节(模式0)④、交换一个字节(模式1)⑤、交换一个字节(模式2)⑥、交换一个字节(模式3)Ⅲ、SPI时序①、发送指令②、指定地址写③、指......
  • Java初学者笔记-04、异常与泛型
    异常异常代表程序出现的问题。Error错误和Exception异常。RuntimeException运行时异常。编译时异常,提醒程序员这里的程序很容易出错。异常的基础处理抛出给上层调用者。使用try-catch处理。异常的处理方案底层异常抛出,最外层捕获异常记录异常并响应合适信息。(少见)最......
  • [数据结构学习笔记15] 汉诺塔(Towers of Hanoi)
    汉诺塔是个古老的游戏,它可以用递归来解决。 关于汉诺塔的玩法和介绍,请参考这里。算法思想:1.目标是把最底下,最大的盘从起始柱子移到终点柱子2.那我们要先把除了最大的盘的其他盘子从起始柱子移到临时柱子上3.然后把最大的盘子从起始柱子移到终点柱子4.把除了最大盘的其......
  • js逆向笔记 绕过某网站开发者工具检测
    js逆向笔记绕过某网站开发者工具检测在这篇博客中,我将分享我在逆向分析爱企查时的一些发现与绕过技巧。最开始,我是偶然发现了这个网站,它在正常使用浏览器按下F12打开开发者工具时,似乎有某种方式禁用了开发者工具。不过,我没有放弃,继续从浏览器的右上角点击手动打开开发者......
  • 【3DGS (1) 】3D Gaussian Splatting全解 (原理+代码+公式) - 笔记
    文章目录1-什么是splatting?2-Splatting的流程3-为什么3dgaussian:是椭球?4-各向异性和各向同性是什么意思?5-`协方差矩阵`怎么就能控制椭球形状呢?6-协方差矩阵怎么就能用旋转和缩放矩阵表达?7-仿射变换本文为B站3DGS讲解视频-【1】捏雪球的文字笔记,以及个......
  • 计算几何~三角形面积、点在三角形内、线段相交代码笔记
    多边形面积的基本公式:鞋带公式。强调多边形点集是按顺序存储;三角形面积基本公式:海伦公式;向量叉积公式;拓扑关系判断:判断点是否在三角形内;判断两条线段是否相交;代码笔记:#pragmaonce#include<iostream>#include<vector>#include<algorithm>#include<cmath>#in......
  • THREE.js学习笔记8——Textures
    这个小节主要学习纹理,Texture纹理是覆盖几何形状表面的图像,不同类型的纹理具有多种不同的效果。这些纹理(尤其是金属性和粗糙度)遵循PBR原则基于物理的渲染许多技术往往遵循现实生活中的方向以获得现实的结果成为现实渲染的标准许多软件、引擎和库都在使用它如何加载纹理?......