算法模板(2):数据结构(3) 复杂数据结构1

news2025/1/19 3:06:48

复杂数据结构(1)

1. Splay

基本概念

什么是 Splay

Splay 是一种二叉查找树,它通过不断将某个节点旋转到根节点,使得整棵树仍然满足二叉查找树的性质,并且保持平衡而不至于退化为链.

旋转操作

为了使 Splay 保持平衡而进行旋转操作,旋转的本质是将某个节点上移一个位置。

旋转需要保证

  • 整棵 Splay 的中序遍历不变(不能破坏二叉查找树的性质)。
  • 受影响的节点维护的信息依然正确有效。
  • root 必须指向旋转后的根节点。

在 Splay 中旋转分为两种:左旋和右旋。

在这里插入图片描述

具体分析旋转步骤 (假设需要旋转的节点为 x x x,其父亲为 y y y,以右旋为例)

  1. y y y 的左儿子指向 x x x 的右儿子,且 x x x 的右儿子的父亲指向 y y ych[y][0]=ch[x][1]; fa[ch[x][1]]=y;
  2. 将 的右儿子指向 ,且 的父亲指向 。 ch[x][chk^1]=y; fa[y]=x;
  3. 如果原来的 还有父亲 ,那么把 的某个儿子(原来 所在的儿子位置)指向 ,且 的父亲指向 。 fa[x]=z; if(z) ch[z][y==ch[z][1]]=x;

Splay 操作

Splay 规定:每访问一个节点后都要强制将其旋转到根节点。此时旋转操作具体分为 6 6 6 种情况讨论(其中 x x x 为需要旋转到根的节点)

  • 如果 x x x 的父亲是根节点,直接将 x x x 左旋或右旋(图 1 , 2 1,2 1,2)。

在这里插入图片描述

  • 如果 x x x 的父亲不是根节点,且 x x x 和父亲的儿子类型相同,首先将其父亲左旋或右旋,然后将 x x x 右旋或左旋(图 3 , 4 3,4 3,4)。
  • 如果 x x x 的父亲不是根节点,且 x x x 和父亲的儿子类型不同,将 x x x 左旋再右旋、或者右旋再左旋(图 5 , 6 5,6 5,6)。
  • Splay 不一定能保证左子树都小于根节点,右子树都大于根节点;但是可以保证中序遍历的顺序就是原序列的顺序。因为 Splay 可以像线段树一样维护一个无序序列。当然,如果该序列本身是有序的,那么 Splay 就是一个平衡二叉树,左子树小于根节点,右子树大于根节点。
  • rotate 示意图
    在这里插入图片描述

950. 郁闷的出纳员

  • 题意:n表示下面有多少条命令,min表示工资下界。有四种命令
  1. I 命令,格式为 I_k,表示新建一个工资档案,初始工资为 k k k。如果某员工的初始工资低于工资下界,他将立刻离开公司. 即离开公司的员工的总数不算这个.
  2. A 命令,格式为 A_k,表示把每位员工的工资加上 k k k.
  3. S 命令,格式为 S_k,表示把每位员工的工资扣除 k k k.
  4. F 命令,格式为 F_k,表示查询第 k k k 多的工资.
  • 对于每条 F 命令,你的程序要输出一行,仅包含一个整数,为当前工资第 k k k 多的员工所拿的工资数,如果 k k k 大于目前员工的数目,则输出-1。输出文件的最后一行包含一个整数,为离开公司的员工的总数。
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int maxn = 100010, INF = 1e9;
int N, M;
struct node {
	int s[2], p, v;
	int size;
	void init(int v_, int p_) {
		v = v_, p = p_;
		size = 1;
	}
}tr[maxn];
int root, idx;
int tot, delta;

void pushup(int x) {
	tr[x].size = tr[tr[x].s[0]].size + tr[tr[x].s[1]].size + 1;
}

void rotate(int x) {
	int y = tr[x].p, z = tr[y].p;
	int k = (tr[y].s[1] == x);
	tr[z].s[tr[z].s[1] == y] = x, tr[x].p = z;
	tr[y].s[k] = tr[x].s[k ^ 1], tr[tr[x].s[k ^ 1]].p = y;
	tr[x].s[k ^ 1] = y, tr[y].p = x;
	pushup(y), pushup(x);
}

void splay(int x, int k) {
	while (tr[x].p != k) {
		int y = tr[x].p, z = tr[y].p;
		if (z != k) {
			if ((tr[y].s[1] == x) ^ (tr[z].s[1] == y)) rotate(x);
			else rotate(y);
		}
		rotate(x);
	}
	if (!k) root = x;
}

int insert(int v) {
	int u = root, p = 0;
	while (u) p = u, u = tr[u].s[v > tr[u].v];
	u = ++idx;
	if (p) tr[p].s[v > tr[p].v] = u;
	tr[u].init(v, p);
	splay(u, 0);
	return u;
}
//大于等于v的最小的节点的编号
int get(int v) {
	int u = root, res;
	while (u) {
		if (tr[u].v >= v) res = u, u = tr[u].s[0];
		else u = tr[u].s[1];
	}
	return res;
}
//找到这个有序序列第 k 个数(算上两个哨兵)
int get_k(int k) {
	int u = root;
	while (u) {
		if (tr[tr[u].s[0]].size >= k) u = tr[u].s[0];
		else if (tr[tr[u].s[0]].size + 1 == k) return tr[u].v;
		else k -= tr[tr[u].s[0]].size + 1, u = tr[u].s[1];
	}
	return -1;  //找不到
}
int main() {
	scanf("%d%d", &N, &M);
	int L = insert(-INF), R = insert(INF);
	while (N--) {
		char op[2];
		int k;
		scanf("%s%d", op, &k);
		if (op[0] == 'I') {
			if (k >= M) {
				//k 需要减 delta,因为默认的是所有人的工资都加上了delta
				k -= delta, insert(k), tot++;
			}
		}
		else if (op[0] == 'A') {
			delta += k;
		}
		else if (op[0] == 'S') {
			delta -= k;
			R = get(M - delta);
			splay(R, 0), splay(L, R);
			tr[L].s[1] = 0;
			pushup(L), pushup(R);
		}
		else {
			if (tr[root].size - 2 < k) printf("-1\n");
			else printf("%d\n", get_k(tr[root].size - k) + delta);
		}
	}
	printf("%d\n", tot - (tr[root].size - 2));
	return 0;
}

2. 树套树

2488. 树套树-简单版

  • 题意:请你写出一种数据结构,来维护一个长度为 n n n 的序列,其中需要提供以下操作:1 pos x,将 p o s pos pos 位置的数修改为 x x x2 l r x,查询整数 x x x 在区间 [ l , r ] [l,r] [l,r] 内的前驱(前驱定义为小于 x x x,且最大的数)。
  • 线段树套 set.
#include<iostream>
#include<algorithm>
#include<cstring>
#include<set>
using namespace std;
const int maxn = 50010, maxm = maxn * 4, INF = 1e9;
int N, M;
struct Tree {
	int l, r;
	multiset<int> s;
}tr[maxm];
int w[maxn];
void build(int u, int l, int r) {
	tr[u] = { l, r };
	//加入两个哨兵
	tr[u].s.insert(-INF), tr[u].s.insert(INF);
	for (int i = l; i <= r; i++) {
		tr[u].s.insert(w[i]);
	}
	if (l == r) return;
	int mid = (l + r) / 2;
	build(2 * u, l, mid), build(2 * u + 1, mid + 1, r);
}
void change(int u, int pos, int x) {
	//这里一定要写成 find,不然会把所有 w[p] 删掉
	tr[u].s.erase(tr[u].s.find(w[pos]));
	tr[u].s.insert(x);
	if (tr[u].l == tr[u].r) return;
	int mid = (tr[u].l + tr[u].r) / 2;
	if (pos <= mid) change(u * 2, pos, x);
	else change(u * 2 + 1, pos, x);
}

int query(int u, int l, int r, int x) {
	if (l <= tr[u].l && tr[u].r <= r) {
		auto it = tr[u].s.lower_bound(x);
		--it;
		return *it;
	}
	int mid = (tr[u].l + tr[u].r) / 2, res = -INF;
	if (l <= mid) res = max(res, query(2 * u, l, r, x));
	if (r > mid) res = max(res, query(2 * u + 1, l, r, x));
	return res;
}

int main() {
	scanf("%d%d", &N, &M);
	for (int i = 1; i <= N; i++) scanf("%d", &w[i]);
	build(1, 1, N);
	while (M--) {
		int op;
		scanf("%d", &op);
		if (op == 1) {
			int pos, x;
			scanf("%d%d", &pos, &x);
			change(1, pos, x);
			w[pos] = x;
		}
		else {
			int l, r, x;
			scanf("%d%d%d", &l, &r, &x);
			printf("%d\n", query(1, l, r, x));
		}
	}
	return 0;
}

2476. 树套树

  • 请你写出一种数据结构,来维护一个长度为 n n n 的数列,其中需要提供以下操作:

1 l r x,查询整数 x x x 在区间 [ l , r ] [l,r] [l,r] 内的排名。

2 l r k,查询区间 [ l , r ] [l,r] [l,r] 内排名为 k k k 的值。

3 pos x,将 p o s pos pos 位置的数修改为 x x x

4 l r x,查询整数 x x x 在区间 [ l , r ] [l,r] [l,r] 内的前驱(前驱定义为小于 x x x,且最大的数)。

5 l r x,查询整数 x x x 在区间 [ l , r ] [l,r] [l,r] 内的后继(后继定义为大于 x x x,且最小的数)。

  • 线段树套splay(思路是比较简单的,看代码就行)
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn = 1500010, INF = 1e9;
//50000 * 4 * 2 + n * logn 两个哨兵
int N, M;

struct node {
	int s[2], p, v;
	int size;
	void init(int _v, int _p) {
		v = _v, p = _p;
		size = 1;
	}
}tr[maxn];

// 为了防止变量名重复,这里的线段树用数组来表示
int L[maxn], R[maxn], T[maxn], idx;
int w[maxn];

//平衡树部分
void pushup(int u) {
	tr[u].size = tr[tr[u].s[0]].size + tr[tr[u].s[1]].size + 1;
}

void rotate(int x) {
	int y = tr[x].p, z = tr[y].p;
	int k = (tr[y].s[1] == x);
	tr[z].s[tr[z].s[1] == y] = x, tr[x].p = z;
	tr[y].s[k] = tr[x].s[k ^ 1], tr[tr[x].s[k ^ 1]].p = y;
	tr[x].s[k ^ 1] = y, tr[y].p = x;
	pushup(y), pushup(x);
}

void splay(int& root, int x, int k) {
	while (tr[x].p != k) {
		int y = tr[x].p, z = tr[y].p;
		if (z != k) {
			if ((tr[y].s[1] == x) ^ (tr[z].s[1] == y)) rotate(x);
			else rotate(y);
		}
		rotate(x);
	}
	if (!k) root = x;
}

void insert(int& root, int v) {
	int u = root, p = 0;
	while (u) p = u, u = tr[u].s[v > tr[u].v];
	u = ++idx;
	if (p) tr[p].s[v > tr[p].v] = u;
	tr[u].init(v, p);
	splay(root, u, 0);
}
//返回以 T[u] 为根的小于等于 x 的数的个数
int get_k(int root, int v) {
	int u = root, res = 0;
	while (u) {
		if (tr[u].v < v) res += tr[tr[u].s[0]].size + 1, u = tr[u].s[1];
		else u = tr[u].s[0];
	}
	return res;
}
//把以 root 为根的splay,删掉x加上y
void update(int& root, int x, int y) {
	int u = root;
	while (u) {
		if (tr[u].v == x) break;
		else if (tr[u].v < x) u = tr[u].s[1];
		else u = tr[u].s[0];
	}
	splay(root, u, 0);
	//删掉平衡树的结点嘛,找到 u 的前驱和后继。
	int l = tr[u].s[0], r = tr[u].s[1];
	while (tr[l].s[1]) l = tr[l].s[1];
	while (tr[r].s[0]) r = tr[r].s[0];
	splay(root, l, 0), splay(root, r, l);
	tr[r].s[0] = 0;
	pushup(r), pushup(l);
	insert(root, y);
}

//前驱
int get_pre(int root, int v) {
	int u = root, res = -INF;
	while (u) {
		if (tr[u].v < v) res = max(res, tr[u].v), u = tr[u].s[1];
		else u = tr[u].s[0];
	}
	return res;
}

int get_suc(int root, int v) {
	int u = root, res = INF;
	while (u) {
		if (tr[u].v > v) res = min(res, tr[u].v), u = tr[u].s[0];
		else u = tr[u].s[1];
	}
	return res;
}

// 线段树部分

void build(int u, int l, int r) {
	L[u] = l, R[u] = r;
	insert(T[u], -INF), insert(T[u], INF);
	for (int i = l; i <= r; i++) insert(T[u], w[i]);
	if (l == r) return;
	int mid = (l + r) / 2;
	build(2 * u, l, mid), build(2 * u + 1, mid + 1, r);
}

int query(int u, int a, int b, int x) {
	if (a <= L[u] && R[u] <= b) {
		//返回以 T[u] 为根的小于 x 的数的个数
		return get_k(T[u], x) - 1;
	}
	int mid = (L[u] + R[u]) / 2, res = 0;
	if (a <= mid) res += query(2 * u, a, b, x);
	if (b > mid) res += query(2 * u + 1, a, b, x);
	return res;
}

void change(int u, int p, int x) {
	update(T[u], w[p], x);
	if (L[u] == R[u]) return;
	int mid = (L[u] + R[u]) / 2;
	if (p <= mid) change(2 * u, p, x);
	else change(2 * u + 1, p, x);
}

int query_pre(int u, int a, int b, int x) {
	//找到以 T[u] 为根节点splay里面小于x的最大的数
	if (a <= L[u] && R[u] <= b) return get_pre(T[u], x);
	int mid = (L[u] + R[u]) / 2, res = -INF;
	if (a <= mid) res = max(res, query_pre(2 * u, a, b, x));
	if (b > mid) res = max(res, query_pre(2 * u + 1, a, b, x));
	return res;
}
int query_suc(int u, int a, int b, int x) {
	//找到以 T[u] 为根节点splay里面大于x的最小的数
	if (a <= L[u] && R[u] <= b) return get_suc(T[u], x);
	int mid = (L[u] + R[u]) / 2, res = INF;
	if (a <= mid) res = min(res, query_suc(2 * u, a, b, x));
	if (b > mid) res = min(res, query_suc(2 * u + 1, a, b, x));
	return res;
}

int main() {
	scanf("%d%d", &N, &M);
	for (int i = 1; i <= N; i++) {
		scanf("%d", &w[i]);
	}
	build(1, 1, N);
	while (M--) {
		int op, a, b, x;
		scanf("%d", &op);
		if (op == 1) {
			//查询x的排名
			scanf("%d%d%d", &a, &b, &x);
			//query求的是小于 x 的个数
			printf("%d\n", query(1, a, b, x) + 1);
		}
		else if (op == 2) {
			//查询排名为 x 的值
			scanf("%d%d%d", &a, &b, &x);
			int l = 0, r = 1e8;
			while (l < r) {
				//最后让答案等于 r,所以向上取整
				int mid = (l + r + 1) / 2;
				if (query(1, a, b, mid) + 1 <= x) l = mid;
				else r = mid - 1;
			}
			printf("%d\n", r);
		}
		else if (op == 3) {
			//修改第pos的值为x
			scanf("%d%d", &a, &x);
			change(1, a, x);
			w[a] = x;
		}
		else if (op == 4) {
			//查询 x 的前驱
			scanf("%d%d%d", &a, &b, &x);
			printf("%d\n", query_pre(1, a, b, x));
		}
		else {
			//查询 x 的后继
			scanf("%d%d%d", &a, &b, &x);
			printf("%d\n", query_suc(1, a, b, x));
		}
	}
	return 0;
}

3. 分块

3.1 分块之基本思想

在这里插入图片描述

243. 一个简单的整数问题2

  • 题意:给定一个长度为 N N N 的数列 A A A,以及 M M M 条指令,每条指令可能是以下两种之一:C l r d,表示把 A [ l ] , A [ l + 1 ] , … , A [ r ] A[l],A[l+1],…,A[r] A[l],A[l+1],,A[r] 都加上 d d dQ l r,表示询问数列中第 l ∼ r l \sim r lr 个数的和。

在这里插入图片描述

#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
const int maxn = 100010, maxm = 350;
typedef long long ll;
ll add[maxm], sum[maxm], w[maxn];
int N, M, len;

int get(int i) {
	//看看映射到了哪一块儿,注意这种写法会让 1~i-1在第一段
	return i / len;
}

void change(int l, int r, int d) {
	if (get(l) == get(r)) {  //段内直接暴力
		for (int i = l; i <= r; i++) w[i] += d, sum[get(i)] += d;
	}
	else {
		int i = l, j = r;
		while (get(i) == get(l)) w[i] += d, sum[get(i)] += d, i++;
		while (get(j) == get(r)) w[j] += d, sum[get(j)] += d, j--;
		for (int k = get(i); k <= get(j); k++) {
			//这个地方,即使长度不足len也没有关系,因为询问的时候,这些长度不足len的段不会作为整体被询问到。
			sum[k] += len * d, add[k] += d;
		}
	}
}

ll query(int l, int r) {
	ll res = 0;
	if (get(l) == get(r)) {
		for (int i = l; i <= r; i++) res += w[i] + add[get(i)];
	}
	else {
		int i = l, j = r;
		while (get(i) == get(l)) res += w[i] + add[get(i)], i++;
		while (get(j) == get(r)) res += w[j] + add[get(j)], j--;
		for (int k = get(i); k <= get(j); k++) res += sum[k];
	}
	return res;
}

int main() {
	scanf("%d%d", &N, &M);
	len = sqrt(N);
	for (int i = 1; i <= N; i++) {
		scanf("%lld", &w[i]);
		sum[get(i)] += w[i];
	}
	char op[2];
	int l, r, d;
	while (M--) {
		scanf("%s%d%d", &op, &l, &r);
		if (op[0] == 'C') {
			scanf("%d", &d);
			change(l, r, d);
		}
		else printf("%lld\n", query(l, r));
	}
	return 0;
}

3.2 分块之块状链表

4. 莫队算法

  • 序列大小为 n n n,询问次数为 m m m,询问维度为 d d d(比如普通莫队的维度为2,即 l l l r r r),那么相当于 m m m 个点均匀分布在维度为 n d n^d nd 的超立方体中,我们的目标就是找到一条哈密顿路,让路径总长度尽可能小一些. 为了方便起见,我们考虑在一个维度上面( d d d 个维度不过是乘了 d \sqrt d d ),点的个数大概是 O ( m 1 d ) O(m^{\frac{1}{d}}) O(md1) 个,因此在一个维度上面的相邻两点之间的距离为 n m 1 d \frac{n}{m^{\frac{1}{d}}} md1n. 哈密顿路径长度为 n m d − 1 d nm^{\frac{d-1}{d}} nmdd1
  • 就是我们要选择的块的大小 n m 1 d \frac{n}{m^{\frac{1}{d}}} md1n,整个算法的复杂度是 O ( n m d − 1 d ) O(nm^{\frac{d-1}{d}}) O(nmdd1)

4.1 莫队之基础莫队

  • 据说奇数块儿升序,偶数块儿降序,这样子有些时候会快一些。
  • 莫队算法开方的时候,千万别忘先强制类型转换为 d o u b l e double double,否则可能会整数溢出.

2492. HH的项链

  • 给一个序列,给很多次询问,求某个区间不同数字的个数。

  • n n n 个数分成 n 2 m \sqrt{\frac{n^2}{m}} mn2 块。对所有询问进行排序,排序标准:

    1. $Q[i].left /len < Q[j].left / len $(块号优先排序)
    2. 如果1相同,则 Q [ i ] . r i g h t < Q [ j ] . r i g h t Q[i].right < Q[j].right Q[i].right<Q[j].right (按照查询的右边界排序)
    3. 关于 i i i j j j 到底怎么变化的问题,只需要记住, a d d add add 是下一个位置的数字, s u b sub sub 是上一个位置的数字.
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
int N, M, len;
const int maxn = 50010, maxm = 200010, maxs = 1000010;

struct Query 
{
	int id, l, r;
}q[maxm];
int w[maxn], cnt[maxs], ans[maxm];

int get(int x) 
{
	return x / len;
}

bool cmp(const Query& a, const Query& b) 
{
	int i = get(a.l), j = get(b.l);
	if (i != j) return i < j;
	return a.r < b.r;
}

void add(int x, int& res) 
{
	if (cnt[x] == 0) res++;
	cnt[x]++;
}
void del(int x, int& res) 
{
	cnt[x]--;
	if (cnt[x] == 0) res--;
}

int main() {
	scanf("%d", &N);
	for (int i = 1; i <= N; i++) scanf("%d", &w[i]);
	scanf("%d", &M);
	len = sqrt((double)N * N / (double)M);
	for (int i = 0; i < M; i++) 
    {
		int l, r;
		scanf("%d%d", &l, &r);
		q[i] = { i, l, r };
	}
	sort(q, q + M, cmp);
	int res = 0;
	// i 指向前一个位置
	for (int k = 0, i = 0, j = 1; k < M; k++) 
    {
        //这里 i 是右端点 j 是左端点
		int id = q[k].id, l = q[k].l, r = q[k].r;
		while (i < r) add(w[++i], res);
		while (i > r) del(w[i--], res);
		while (j < l) del(w[j++], res);
		while (j > l) add(w[--j], res);
		ans[id] = res;
	}
	for (int i = 0; i < M; i++) printf("%d\n", ans[i]);
	return 0;
}
树状数组做法
  • 用树状数组维护。我们只关心右边的那个数字。比如 1 3 2 1,那么第一个 1 就不重要。我们就可以把第一个1减掉,把最后一个1加上。
  • 具体思路是这样的,我们按照右端点对询问进行排序,然后让一个指针开始往右扫描,每次都记录每个颜色最后出现的位置,把那个位置的值置为1,假如这个颜色在之前出现过,就把那个位置置为0. 然后每次询问都查询 [ l , r ] [l,r] [l,r] 区间有多少个1. 这就是单点修改,区间查询问题,用树状数组维护.
#include<bits/stdc++.h>
using namespace std;
const int N = 50010, M = 200010;

int tr[N], last[1000010], a[N], ans[M];
int n, m;

struct P
{
    int l, r, id;
    bool operator < (const P& p) const
    {
        return r < p.r;
    }
}Q[M];

int lowbit(int x)
{
    return x & -x;
}

void add(int x, int c)
{
    for(int i = x; i <= n; i += lowbit(i))
    {
        tr[i] += c;
    }
}

int sum(int x)
{
    int res = 0;
    for(int i = x; i; i -= lowbit(i)) res += tr[i];
    return res;
}

int main()
{
    scanf("%d", &n);
    for(int i = 1; i <= n; i++)
    {
        scanf("%d", &a[i]);
    }
    scanf("%d", &m);
    for(int i = 1; i <= m; i++)
    {
        scanf("%d%d", &Q[i].l, &Q[i].r);
        Q[i].id = i;
    }
    sort(Q + 1, Q + m + 1);

    for(int i = 1, j = 1; i <= m; i++)
    {
        for(; j <= Q[i].r; j++)
        {
            if(last[a[j]]) add(last[a[j]], -1);
            add(j, 1), last[a[j]] = j;
        }
        ans[Q[i].id] = sum(Q[i].r) - sum(Q[i].l - 1);
    }
    for(int i = 1; i <= m; i++) printf("%d\n", ans[i]);
    return 0;
}

4.2 莫队之带修改的莫队

2521. 数颜色

  • 题意:墨墨购买了一套 N N N 支彩色画笔(其中有些颜色可能相同),摆成一排,你需要回答墨墨的提问。墨墨会像你发布如下指令:Q L R 代表询问你从第 L L L 支画笔到第 R R R 支画笔中共有几种不同颜色的画笔。R P Col 把第 P P P 支画笔替换为颜色 C o l Col Col
  • 输入: 1 ≤ N , M ≤ 10000 1\le N,M\le 10000 1N,M10000,修改操作不多于 1000 1000 1000 次,所有的输入数据中出现的所有整数均大于等于 1 1 1 且不超过 1 0 6 10^6 106
  • 设序列大小为 n n n,询问次数是 m m m,修改次数是 m m mt块的大小是 a a a,由于 m m m 中的 l , r l, r l,r t t t 这三个询问的维度并不相等。因此分块大小应该这样计算: l l l 移动次数为 O ( a m + n ) O(am+n) O(am+n) r r r 移动次数是 O ( a m + n 2 a ) O(am+\frac{n^2}{a}) O(am+an2) t t t 的移动次数是 O ( t n 2 a 2 ) O(t\frac{n^2}{a^2}) O(ta2n2). 然后接下来就是玄学推导。由于题目的 n n n m m m 的范围大致一致,因此 l l l 可以写为 O ( a n ) O(an) O(an) r r r 可以写为 O ( a n ) O(an) O(an),则由 a n = t n 2 a 2 an = t\frac{n^2}{a^2} an=ta2n2 a = n t 3 a = \sqrt[3]{nt} a=3nt
  • 其实就是排序的时候用到了分块儿,之后还是一个暴力做法。
#include<bits/stdc++.h>
using namespace std;
const int N = 10010, M = 1000010;

int a[N], n, m, mc, mq, len;
int res, cnt[M], ans[N];

struct Query
{
    int l, r, id, t;
}Q[N];

struct Modify
{
    int p, c;
}c[N];

int get(int x)
{
    return x / len;
}

bool cmp(const Query& q1, const Query& q2)
{
    int l1 = get(q1.l), l2 = get(q2.l);
    int r1 = get(q1.r), r2 = get(q2.r);
    int t1 = q1.t, t2 = q2.t;

    if(l1 != l2) return l1 < l2;
    if(r1 != r2) return r1 < r2;
    return t1 < t2;
}

void add(int x, int& res)
{
    if(cnt[x] == 0) res++;
    cnt[x]++;
}

void sub(int x, int& res)
{
    cnt[x]--;
    if(cnt[x] == 0) res--;
}

int main()
{
    scanf("%d%d", &n, &m);

    for(int i = 1; i <= n; i++)
    {
        scanf("%d", &a[i]);
    }

    for(int i = 1; i <= m; i++)
    {
        static char op[5];
        static int x, y;
        scanf("%s%d%d", op, &x, &y);
        if(*op == 'Q') mq++, Q[mq] = {x, y, mq, mc};
        else c[++mc] = {x, y};
    }

    int res = 0;

    len = cbrt(1.0 * n * (mc + 1));
    sort(Q + 1, Q + mq + 1, cmp);

    for(int k = 1, i = 1, j = 0, t = 0; k <= mq; k++)
    {
        //这里的 i 表示左端点,j 表示右端点
        //一定要先改变右端点再改变左端点
        int l = Q[k].l, r = Q[k].r, id = Q[k].id, tm = Q[k].t;
        while(j > r) sub(a[j--], res);
        while(j < r) add(a[++j], res);
        while(i < l) sub(a[i++], res);
        while(i > l) add(a[--i], res);

        while(t < tm)
        {
            t++;
            if(l <= c[t].p && c[t].p <= r)
            {
                sub(a[c[t].p], res);
                add(c[t].c, res);
            }
            swap(a[c[t].p], c[t].c);
        }
        while(t > tm)
        {
            if(l <= c[t].p && c[t].p <= r)
            {
                sub(a[c[t].p], res);
                add(c[t].c, res);
            }
            swap(a[c[t].p], c[t].c);
            t--;
        }
        ans[id] = res;
    }
    for(int i = 1; i <= mq; i++)
    {
        printf("%d\n", ans[i]);
    }
    return 0;
}

4.3 莫队之回滚莫队

有些题目在区间转移时,可能会出现增加或者删除无法实现的问题。在只有增加不可实现或者只有删除不可实现的时候,就可以使用回滚莫队在 O ( n n ) O(n \sqrt n) O(nn ) 的时间内解决问题。回滚莫队的核心思想就是既然我只能实现一个操作,那么我就只使用一个操作,剩下的交给回滚解决。

回滚莫队分为只使用增加操作的回滚莫队和只使用删除操作的回滚莫队。以下仅介绍只使用增加操作的回滚莫队,只使用删除操作的回滚莫队和只使用增加操作的回滚莫队只在算法实现上有一点区别,故不再赘述。

2523. 历史研究

  • 给一个序列 n ≤ 1 0 5 n \le 10^5 n105,序列由一系列数字 X i ≤ 1 0 9 X_i \le 10^9 Xi109 组成,下面有一系列询问 m ≤ 1 0 5 m \le 10^5 m105,每次询问一个区间 [ l , r ] [l,r] [l,r],求这个区间这个数的最大值:区间每个数出现次数 乘 这个数字.
  • 这个题就是典型的增加方便删除困难,因为删除操作维护最大值不容易实现。这道题还有一个地方就是卡常,离散化用 vector 实现,更快一些.
  1. 首先分块排序,第一关键字按照左端点 l l l 所在块升序排序,第二关键字按照右端点升序排序.
  2. 对于左右端点在同一块内的询问可以暴力处理
  3. 左右端点横跨两个块的询问(记为块 A A A A A A 之外),初始时将右端点置于块 A A A 的右端点,左端点置于块 A A A 的右端点加 1 1 1 的位置,右端点只会一直往右左,暴力维护左端点即可。
#include<bits/stdc++.h>
using namespace std;
const int N = 100010;

typedef long long ll;
ll a[N], res, ans[N];
int len, idx;

int cnt[N];
vector<ll> nums;

int get_id(ll x)
{
    return lower_bound(nums.begin(), nums.end(), x) - nums.begin();
}

struct Query
{
    int l, r, id;
}Q[N];

void add(int x)
{
    cnt[x]++;
    if(cnt[x] * nums[x] > res) res = cnt[x] * nums[x];
}

void sub(int x)
{
    cnt[x]--;
}

int get(int x)
{
    return x / len;
}

bool cmp(Query& q1, Query& q2)
{
    int l1 = get(q1.l), l2 = get(q2.l);
    if(l1 != l2) return l1 < l2;
    return q1.r < q2.r;
}

int main()
{
    int n, m;
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= n; i++)
    {
        scanf("%lld", &a[i]);
        nums.push_back(a[i]);
    }

    sort(nums.begin(), nums.end());
    nums.erase(unique(nums.begin(), nums.end()), nums.end());
    //这样子,a中存储的就是每一个数字的相对大小了.
    for(int i = 1; i <= n; i++)
    {
        a[i] = get_id(a[i]);
    }
    
    for(int i = 1; i <= m; i++)
    {
        scanf("%d%d", &Q[i].l, &Q[i].r);
        Q[i].id = i;
    }

    len = sqrt(m);
    sort(Q + 1, Q + m + 1, cmp);

    for(int x = 1; x <= m; )
    {
        memset(cnt, 0, sizeof cnt);
        int y = x;
        while(y <= m && get(Q[x].l) == get(Q[y].l)) y++;
        
        while(x < y)
        {
            res = 0;
            if(get(Q[x].l) != get(Q[x].r)) break;
            int l = Q[x].l, r = Q[x].r, id = Q[x].id;
            for(int k = l; k <= r; k++) add(a[k]);
            ans[id] = res;
            for(int k = l; k <= r; k++) sub(a[k]);
            x++;
        }
        
        int right = get(Q[x].l) * len + len - 1;
        int i = right + 1, j = right;
        res = 0;
        while(x < y)
        {
            int l = Q[x].l, r = Q[x].r, id = Q[x].id;
            while(j < r) add(a[++j]);
            ll backup = res;
            while(i > l) add(a[--i]);
            ans[id] = res;
            res = backup;
            while(i < right + 1) sub(a[i++]);
            x++;
        }
    }
    
    for(int i = 1; i <= m; i++) printf("%lld\n", ans[i]);
    return 0;
}

4.4 莫队之树上莫队

2534. 树上计数2

  • 给定一棵 N ( N ≤ 40000 ) N(N \le 40000) N(N40000) 个节点的树,节点编号从 1 1 1 N N N,每个节点都有一个整数权值。现在,我们要进行 M ≤ 1 0 5 M \le 10^5 M105 次询问,格式为 u v,对于每个询问你需要回答从 u u u v v v 的路径上(包括两端点)共有多少种不同的点权值。点的权值在 i n t int int 范围内.

在这里插入图片描述

  1. 先把树的问题转化为序列问题,即求出树的欧拉序列.
  2. 记录每一个结点第一次出现的位置 f i r s t [ u ] first[u] first[u],和最后一次出现的位置 l a s t [ u ] last[u] last[u].
  3. 不妨设 f i r s t [ x ] < f i r s t [ y ] first[x] < first[y] first[x]<first[y]:
    1. l c a ( x , y ) = x lca(x, y) = x lca(x,y)=x,则区间为在欧拉序列 [ f i r s t [ x ] , f i r s t [ y ] ] [first[x], first[y]] [first[x],first[y]] 中只出现了一次的点.
    2. l c a ( x , y ) ≠ x lca(x, y) \ne x lca(x,y)=x,则区间为在欧拉序列 [ l a s t [ x ] , f i r s t [ y ] ] [last[x], first[y]] [last[x],first[y]] 中出现一次的点加上 l c a ( x , y ) lca(x, y) lca(x,y).
#include<bits/stdc++.h>
using namespace std;
const int N = 100010;

int h[N], e[N], ne[N], idx;
int a[N], n, m, ans[N];
int seq[N], top, que[N], first[N], last[N];
vector<int> nums;

int cnt[N], st[N];
int len;
int depth[N], f[N][16];

void add_edge(int a, int b)
{
    e[idx] = b, ne[idx] = h[a], h[a] = idx++;
}

struct Query
{
    int l, r, id, p;
}Q[N];

int get(int x)
{
    return x / len;
}

bool cmp(const Query& q1, const Query& q2)
{
    int l1 = get(q1.l), l2 = get(q2.l);
    if(l1 != l2) return l1 < l2;
    return q1.r < q2.r;
}

void dfs(int u, int fa)
{
    seq[++top] = u;
    first[u] = top;
    for(int i = h[u]; i != -1; i = ne[i])
    {
        int v = e[i];
        if(v == fa) continue;
        dfs(v, u);
    }
    seq[++top] = u;
    last[u] = top;
}

void bfs()
{
    memset(depth, 0x3f, sizeof depth);
    depth[0] = 0, depth[1] = 1;
    int hh = 0, tt = -1;
    que[++tt] = 1;
    while(hh <= tt)
    {
        int u = que[hh++];
        for(int i = h[u]; i != -1; i = ne[i])
        {
            int v = e[i];
            if(depth[v] > depth[u] + 1)
            {
                depth[v] = depth[u] + 1;
                f[v][0] = u;
                for(int k = 1; k <= 15; k++) f[v][k] = f[f[v][k - 1]][k - 1];
                que[++tt] = v;
            }
        }
    }
}

int lca(int a, int b)
{
    if(depth[a] < depth[b]) swap(a, b);
    for(int k = 15; k >= 0; k--)
    {
        if(depth[f[a][k]] >= depth[b])
        {
            a = f[a][k];
        }
    }
    if(a == b) return a;

    for(int k = 15; k >= 0; k--)
    {
        if(f[a][k] != f[b][k])
        {
            a = f[a][k];
            b = f[b][k];
        }
    }
    return f[a][0];
}

void add(int x, int& res)
{
    st[x] ^= 1;
    if(st[x])
    {
        cnt[a[x]]++;
        if(cnt[a[x]] == 1) res++;
    }
    else
    {
        cnt[a[x]]--;
        if(!cnt[a[x]]) res--;
    }
}

int main()
{
    memset(h, -1, sizeof h);
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= n; i++)
    {
        scanf("%d", &a[i]);
        nums.push_back(a[i]);
    }

    sort(nums.begin(), nums.end());
    nums.erase(unique(nums.begin(), nums.end()), nums.end());

    for(int i = 1; i <= n; i++)
    {
        a[i] = lower_bound(nums.begin(), nums.end(), a[i]) - nums.begin();
    }

    for(int i = 1; i < n; i++)
    {
        int x, y;
        scanf("%d%d", &x, &y);
        add_edge(x, y), add_edge(y, x);
    }

    dfs(1, -1);
    bfs();

    for(int i = 1; i <= m; i++)
    {
        int x, y;
        scanf("%d%d", &x, &y);
        
        if(first[x] > first[y]) swap(x, y);
        
        int p = lca(x, y);
        if(p == x) Q[i] = {first[x], first[y], i};
        else Q[i] = {last[x], first[y], i, p};
    }

    len = sqrt(top);
    sort(Q + 1, Q + m + 1, cmp);

    int res = 0;
    for(int i = 1, L = 1, R = 0; i <= m; i++)
    {
        int l = Q[i].l, r = Q[i].r, id = Q[i].id, p = Q[i].p;
        while(R < r) add(seq[++R], res);
        while(R > r) add(seq[R--], res);
        while(L < l) add(seq[L++], res);
        while(L > l) add(seq[--L], res);

        if(p) add(p, res);
        ans[id] = res;
        if(p) add(p, res);
    }

    for(int i = 1; i <= m; i++)
    {
        printf("%d\n", ans[i]);
    }
}

4.5 莫队之二次离线莫队

2535. 二次离线莫队

  • 题意:给定一个长度为 n n n 的序列 a 1 , a 2 , . . . , a n a_1,a_2,...,a_n a1,a2,...,an 以及一个整数 k k k。现在要进行 m m m 次询问,每次询问给定一个区间 [ l , r ] [l,r] [l,r],请你求出共有多少个数对 ( i , j ) (i,j) (i,j) 满足 l ≤ i < j ≤ r l \le i < j \le r li<jr a i ⊕ a j a_i \oplus a_j aiaj 的结果在二进制表示下恰好有 k k k 1 1 1 n ≤ 1 0 5 , 0 ≤ k ≤ 14 , 0 ≤ a i < 2 14 n\le 10^5,0\le k \le 14, 0 \le a_i < 2^{14} n105,0k14,0ai<214

5. 树链剖分

将树转化为一个序列,将树中路径转化为长度为 log ⁡ n \log n logn 段连续区间,也叫 log ⁡ n \log n logn 条重链.

基本概念

  • 重儿子:父亲节点的所有儿子中子树结点数目最多(size最大)的结点;
  • 轻儿子:父亲节点中除了重儿子以外的儿子;
  • 重边:父亲结点和重儿子连成的边;
  • 轻边:父亲节点和轻儿子连成的边;
  • 重链:由多条重边连接而成的路径;
  • 轻链:由多条轻边连接而成的路径;

性质

**对于节点数为 n n n 的树,从任意节点向上走到根节点,经过的轻边数量不会超过 log ⁡ n \log n logn. ** 这是因为当我们向下经过一条 轻边 时,所在子树的大小至少会除以二。所以说,对于树上的任意一条路径,把它拆分成从 l c a lca lca 分别向两边往下走,分别最多走 O ( log ⁡ n ) O(\log n) O(logn) 次,树上的每条路径都可以被拆分成不超过 O ( log ⁡ n ) O(\log n) O(logn) 条重链。

对于每个轻儿子,如果该结点不是叶节点,那么他一定是重链的开始。

树上每个节点都属于且仅属于一条重链

在这里插入图片描述

按照 D F S DFS DFS 序生成序列. 先访问重儿子,再访问轻儿子. 标红的点是重儿子,标红的边是重边.

一些定义

f a ( x ) fa(x) fa(x) 表示节点 x x x 在树上的父亲。

d e p ( x ) dep(x) dep(x) 表示节点 x x x 在树上的深度。

s z ( x ) sz(x) sz(x) 表示节点 x x x 的子树的节点个数。

s o n ( x ) son(x) son(x) 表示节点 x x x重儿子 ,即所有儿子中子树大小最大的一个。

定义 重边 表示连接两个重儿子的边。

定义 重路径 表示重边连成的一条链。

t o p ( x ) top(x) top(x) 表示节点 所在 重路径 的顶部节点(深度最小)。 t o p ( x ) top(x) top(x) 一定是一个轻儿子。

i d ( x ) id(x) id(x) 表示节点 x x x时间戳 ,也是其在线段树中的编号。

n w ( x ) nw(x) nw(x) 表示时间戳为 x x x 的结点的权值。

我们进行两遍 DFS 预处理出这些值,其中第一次 DFS 求出 f a ( x ) , d e p ( x ) , s z ( x ) , s o n ( x ) fa(x),dep(x),sz(x),son(x) fa(x),dep(x),sz(x),son(x),第二次 DFS 求出 t o p ( x ) , i d ( x ) top(x),id(x) top(x),id(x)

2568. 树链剖分

给定一棵树,树中包含 n n n 个节点(编号 1 ∼ n 1 \sim n 1n),其中第 i i i 个节点的权值为 a i a_i ai。初始时, 1 1 1 号节点为树的根节点。现在要对该树进行 m m m 次操作,操作分为以下 4 4 4 种类型:

1 u v k,修改路径上节点权值,将节点 u u u 和节点 v v v 之间路径上的所有节点(包括这两个节点)的权值增加 k k k

2 u k,修改子树上节点权值,将以节点 u u u 为根的子树上的所有节点的权值增加 k k k

3 u v,询问路径,询问节点 u u u 和节点 v v v 之间路径上的所有节点(包括这两个节点)的权值和。

4 u,询问子树,询问以节点 u u u 为根的子树上的所有节点的权值和。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

const int N = 100010, M = N * 2;
int w[N], h[N], e[M], ne[M], idx;
int n, m;

void add(int a, int b)
{
    e[idx] = b, ne[idx] = h[a], h[a] = idx++;
}

int dep[N], sz[N], fa[N], son[N];
int nw[N], id[N], top[N], cnt;

void dfs1(int u, int father, int depth)
{
    dep[u] = depth, sz[u] = 1, fa[u] = father;
    for(int i = h[u]; i != -1; i = ne[i])
    {
        int v = e[i];
        if(v == father) continue;
        dfs1(v, u, depth + 1);
        if(sz[son[u]] < sz[v]) son[u] = v;
        sz[u] += sz[v];
    }
}

//t 表示当前结点所在重链的开头结点是什么
void dfs2(int u, int t)
{
    id[u] = ++cnt; nw[cnt] = w[u], top[u] = t;
    if(!son[u]) return;
    dfs2(son[u], t);
    for(int i = h[u]; i != -1; i = ne[i])
    {
        int v = e[i];
        //判断 v 是否为 father 时,千万别写错,不然可能会 MLE
        if(v == fa[u] || son[u] == v) continue;
        dfs2(v, v);
    }
}

struct node
{
    int l, r;
    ll sum, add;
}tr[N * 4];

void pushup(int u)
{
    tr[u].sum = tr[2 * u].sum + tr[2 * u + 1].sum;
}

void pushdown(int u)
{
    auto& root = tr[u], &left = tr[2 * u], &right = tr[2 * u + 1];
    if(root.add)
    {
        left.sum += root.add * (left.r - left.l + 1);
        right.sum += root.add * (right.r - right.l + 1);
        left.add += root.add, right.add += root.add;
        root.add = 0;
    }
}

void build(int u, int l, int r)
{
    tr[u] = {l, r};
    if(l == r) tr[u].sum = nw[l];
    else
    {
        int mid = (l + r) / 2;
        build(2 * u, l, mid), build(2 * u + 1, mid + 1, r);
        pushup(u);
    }
}

void update(int u, int l, int r, int k)
{
    if(l <= tr[u].l && tr[u].r <= r)
    {
        tr[u].add += k;
        tr[u].sum += (ll)k * (tr[u].r - tr[u].l + 1);
    }
    else
    {
        pushdown(u);
        int mid = (tr[u].l + tr[u].r) / 2;
        if(l <= mid) update(2 * u, l, r, k);
        if(r > mid) update(2 * u + 1, l, r, k);
        pushup(u);
    }
}

ll query(int u, int l, int r)
{
    if(l <= tr[u].l && tr[u].r <= r) return tr[u].sum;
    pushdown(u);
    int mid = (tr[u].l + tr[u].r) / 2;
    ll res = 0;
    if(l <= mid) res += query(2 * u, l, r);
    if(r > mid) res += query(2 * u + 1, l, r);
    return res;
}

void update_path(int u, int v, int k)
{
    while(top[u] != top[v])
    {
        if(dep[top[u]] < dep[top[v]]) swap(u, v);
        //传进线段树的下标s结点编号。
        update(1, id[top[u]], id[u], k);
        u = fa[top[u]];
    }
    if(dep[u] < dep[v]) swap(u, v);
    update(1, id[v], id[u], k);
}

void query_path(int u, int v)
{
    ll res = 0;
    while(top[u] != top[v])
    {
        if(dep[top[u]] < dep[top[v]]) swap(u, v);
        res += query(1, id[top[u]], id[u]);
        u = fa[top[u]];
    }
    if(dep[u] < dep[v]) swap(u, v);
    res += query(1, id[v], id[u]);
    printf("%lld\n", res);
}

void update_tree(int u, int k)
{
    update(1, id[u], id[u] + sz[u] - 1, k);
}

void query_tree(int u)
{
    printf("%lld\n", query(1, id[u], id[u] + sz[u] - 1));
}

int main()
{
    scanf("%d", &n);
    for (int i = 1; i <= n; i ++ ) scanf("%d", &w[i]);
    memset(h, -1, sizeof h);
    for (int i = 0; i < n - 1; i ++ )
    {
        int a, b;
        scanf("%d%d", &a, &b);
        add(a, b), add(b, a);
    }
    dfs1(1, -1, 1);
    dfs2(1, 1);
    build(1, 1, n);

    scanf("%d", &m);
    while (m -- )
    {
        int t, u, v, k;
        scanf("%d%d", &t, &u);
        if (t == 1)
        {
            scanf("%d%d", &v, &k);
            update_path(u, v, k);
        }
        else if (t == 2)
        {
            scanf("%d", &k);
            update_tree(u, k);
        }
        else if (t == 3)
        {
            scanf("%d", &v);
            query_path(u, v);
        }
        else query_tree(u);
    }
    return 0;
}

6. 动态树

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/418720.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

虚拟主持人是什么?有哪些应用场景?

虚拟主持人是一种通动作捕捉技术或人工智能等技术&#xff0c;来驱动虚拟形象模拟真实主持人的动作、声音和表情&#xff0c;为各种场合提供主持服务的数字化角色。虚拟主持人的外貌、服装、发型、口音等都可以根据客户的需求和喜好进行定制&#xff0c;甚至可以创建出具有明星…

冶炼金属(十四届蓝桥杯B组)(超级详细)

目录 1、冶炼金属 解法一&#xff1a;(二分算法) 解法二&#xff1a;&#xff08;数学公式推导&#xff09; 2、永远不要试图用拖延来期待事情发生改变&#xff01;&#xff01;&#xff01; 1、冶炼金属 小蓝有一个神奇的炉子用于将普通金属 O冶炼成为一种特殊金属 X。 这…

OceanBase 4.1 发版 | 一个面向开发者的里程碑版本

欢迎访问 OceanBase 官网获取更多信息&#xff1a;https://www.oceanbase.com/ 2022 年 8 月&#xff0c;OceanBase发布了 4.0 版本&#xff08;小鱼&#xff09;&#xff0c;作为业内首个单机分布式一体化架构&#xff0c;兼顾了分布式架构的扩展性和集中式架构的性能优势&…

Iptables安全防火墙企业实战

1. IPtables入门简介 Netfilter/Iptables(以下简称Iptables)是unix/linux自带的一款优秀且开放源代码的完全自由的基于包过滤的防火墙工具&#xff0c;它的功能十分强大&#xff0c;使用非常灵活&#xff0c;可以对流入和流出服务器的数据包进行很精细的控制。Iptables主要工作…

用通俗易懂的话来解释什么是Socket

前言 我在去年就学习过Java中Socket的使用&#xff0c;但对于Socket的理解一直都是迷迷糊糊的。看了网上很多关于Socket的介绍&#xff0c;看完还是不太理解到底什么是Socket&#xff0c;还是很迷。直到最近在学习计算机网络&#xff0c;我才对Socket有了一个更深地理解。之前…

sqoop数据导出、脚本使用

目录 准备表与数据 数据导出 脚本调用 准备表与数据 mysql表 CREATE TABLE user (id int(20),name varchar(20) )ENGINEINNODB DEFAULT CHARSETutf8; hive表 create table users( id bigint, name string ) row format delimited fields terminated by "\t";…

消息中间件RabbitMQ---Docker安装RabbitMQ、以及RabbitMQ的基本使用【二】

1、安装 1.1 拉取镜像 docker pull rabbitmq:management1.2 查看镜像 docker images1.3 创建容器 docker run -d --name rabbitmq -p 5671:5671 -p 5672:5672 -p 4369:4369 -p 25672:25672 -p 15671:15671 -p 15672:15672 rabbitmq:management1.4 浏览器访问 登录进去后的效…

【C++】16.红黑树

1.红黑树 红黑树&#xff0c;是一种二叉搜索树&#xff0c;但在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是Red或B lack。 通过对任何一条从根到叶子的路径上各个结点着色方式的限制&#xff0c;红黑树确保没有一条路 径会比其他路径长出俩倍&#xff0c;因而…

UNIAPP实战项目笔记60 前端使用token来验证是否已经登录

UNIAPP实战项目笔记60 前端使用token来验证是否已经登录 获取数据库中用户是否有token值,并存入 store.user.token中,有值代表已经登录 detail页面 通过验证token拦截未登录用户bin跳转到登录页面 实际案例图片 后端接口文件 index.js var express require(express); var rou…

Python-入门基础篇(1)

这里使用的编译器是pycharm 打开pycharm 新建项目 选择一个文件路径&#xff08;文件名字不要带中文&#xff09; 新建一个.py 文件 这里就要编写代码啦 Python入门最基础的一些函数 print() 该函数可以输出内容到控制台。在括号中输入要输出的内容&#xff0c;用逗号隔开…

RocketMQ延迟消息源码分析

写作目的 第一个原因&#xff1a;最近玩哔哩哔哩遇到一个RocketMQ的Contributor&#xff0c;一开始不知道他是Contributor&#xff0c;后来问到延迟消息的时候这块还不是很了解&#xff0c;他告诉我学习要系统&#xff0c;你既然了解事务消息那我理解应该也了解延迟消息&#…

BGP路由实验

要求1 使用 Preval 策略 [r4]ip ip-prefix PV permit 192.168.10.0 24 [r4]route-policy PV permit node 10 [r4-route-policy]if-match ip-prefix PV [r4-route-policy]apply preferred-value 100 [r4]route-policy PV permit node 20 [r4-bgp]peer 24.0.0.2 route-policy…

redis网络模型

用户空间和内核空间IO五种IO模型阻塞IO非阻塞IOIO多路复用selectpollepollweb服务流程信号驱动IO异步IOIO模型比较redis网络模型redis为什么是单线程redis单线程网络模型流程用户空间和内核空间 为安全&#xff0c;将用户应用和系统应用分隔开&#xff0c;产生用户空间和内核空…

thanos prometheus 的高可用、长期存储二进制部署

1.简介 http://thanos.io/ thanos 是具有长期存储功能的开源、高可用性 Prometheus的集群组件。 全局查询视图 跨多个 Prometheus 服务器和集群查询指标 无限保留 使用对象存储扩展系统&#xff0c;不限时间保留指标。 Prometheus兼容 兼容 Prometheus api&#xff0c;用于…

我在windows10下,使用msys64 mingw64终端

系列文章目录 文章目录系列文章目录前言一、MSYS2是什么&#xff1f;前言 msys2官网 MSYS2 &#xff08;Minimal SYStem 2&#xff09; 是一个MSYS的独立改写版本&#xff0c;主要用于 shell 命令行开发环境。 同时它也是一个在Cygwin &#xff08;POSIX 兼容性层&#xff09…

软件测试简历如何编写?还在乱写?精细优化让自己脱颖而出......

目录&#xff1a;导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09;前言 简历是我们向面试官…

教育大数据总体解决方案(6)

触控录播主机集成视频编码、实时导播、音频处理、图像跟踪、电子云台等功能&#xff0c;无需额外的辅助跟踪设备。支持5路视频信号的采集录制、画面自动跟踪及全自动/半自动导播切换&#xff0c;让教师免于分心调控&#xff0c;专注于课堂教学。 整机采取一体化设计及三合一按键…

【数据结构与算法】栈和队列(StackQueue)

TOC 数据结构—栈 栈的概念 要想学习一个东西&#xff0c;概念是一定要看并且理解的&#xff0c;那么栈是个什么玩意呢&#xff1f; 栈&#xff1a;一种特殊的线性表&#xff0c;其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶&#x…

v851s g2d 模块 sample 深究

1. g2d 模块概述 g2d 主要功能: 1)旋转:支持90、180、270旋转; 2)镜像反转:H / V; 3) scale:放缩 4)格式转换:yuv 转 rgb 等,多种格式相互间转换; 5)透明叠加功能:实现两个rgb图片叠加; 6)矩形填充,等诸多功能; 2. g2d 配置 1)源码目录:tina-v853-docker/…

Zabbix自定义监控mysql数据库、自动注册服务器及部署代理服务器

目录 一、zabbix自定义监控数据库 1、编写监控脚本 2、服务端测试 3、web页面配置 ①创建自定义监控项 ②创建触发器 ③创建图形 ④测试自定义监控是否成功 二、zabbix自动注册 1、什么是自动注册 2、环境准备 3、 zabbix客户端配置 4、web页面配置自动注册 5、…