主席树学习整理

主席树学习整理

1、求静态区域第k大

1.1静态主席树

发明者的原话:“对于原序列的每一个前缀[1···i]建立出一棵线段树维护值域上每个数出现的次数,则其树是可减的”

可以加减的理由:主席树的每个节点保存的是一颗线段树,维护的区间信息,结构相同,因此具有可加减性(关键

首先开一个数组t[n],存储内容为a中排序并去重的值(类似于离散化),每棵线段树维护的内容是a1...ai此区间中的树在t[n]中出现的次数

1.2举个栗子

an:4 1 1 2 8 9 4 4 3

将序列排序并去重后得到t[n]:

tn:1 2 3 4 8 9

对前缀a[1...9]建树,1*2,2*1,3*1,4*3,8*1,9*1,每个数出现的次数即为线段树维护的值,树中每个节点表示t[i,j],中的数字在a[1...9]中出现的次数

1.3静态主席树的一些点

1.建树时首先需要建一棵空的线段树,即最原始的主席树,此时主席树只含有一个空的节点,之后依次对原序列按某种顺序更新,就是将原序列加入到相应的位置

2.主席树是一种特殊的线段树集,它包含了所有线段树的优势,并且可以保存历史状态,主席树查找和更新的时空复杂度都为O(nlogn)且总空间复杂度为O(nlogn+nlogn)前者为空树的复杂度,后者为更新n次的空间复杂度,缺点是空间损耗巨大

3.主席树可以处理区间[L,R]中介于[x,y]的值的问题

4.若增加空间垃圾回收则可以使空间复杂度降低一个log

1.4代码

#include <iostream>
#include <cstdio>
#include <algorithm>
using namespace std;
#pragma warning (disable:4996)
const int maxn = 200005;
int n, m, tsize;
int a[maxn], b[maxn], t[maxn], tot = 0;
struct charitree{
	int left, right, size;
}tree[maxn * 30];
int build(int stdl, int stdr) {
	int root = tot++;
	tree[root].size = 0;
	if (stdl == stdr)
		return root;
	int mid = (stdl + stdr) >> 1;
	tree[root].left = build(stdl, mid);
	tree[root].right = build(mid + 1, stdr);
	return root;
}
int update(int root, int x) {
	int now = tot++;
	int tmp = now;
	tree[now].size = tree[root].size + 1;
	int stdl = 1, stdr = tsize;
	while (stdl < stdr) {
		int mid = (stdl + stdr) >> 1;
		if (x <= mid) {
			tree[now].left = tot++;
			tree[now].right = tree[root].right;
			now = tree[now].left;
			stdr = mid;
			root = tree[root].left;
		}
		else {
			tree[now].left = tree[root].left;
			tree[now].right = tot++;
			now = tree[now].right;
			stdl = mid + 1;
			root = tree[root].right;
		}
		tree[now].size = tree[root].size + 1;
	}
	return tmp;
}
int ask(int stdl, int stdr, int k) {
	int l = 1, r = tsize;
	while (l<r)
	{
		int mid = (l + r) >> 1;
		if (tree[tree[stdr].left].size - tree[tree[stdl].left].size >= k) {
			r = mid;
			stdl = tree[stdl].left;
			stdr = tree[stdr].left;
		}
		else {
			l = mid + 1;
			k -= tree[tree[stdr].left].size - tree[tree[stdl].left].size;
			stdl = tree[stdl].right;
			stdr = tree[stdr].right;
		}
	}
	return l;
}
int find(int x) {
	return lower_bound(b + 1, b + 1 + tsize, x) - b;
}
int main() {
	scanf("%d%d", &n, &m);
	for (int i = 1; i <= n; i++) {
		scanf("%d", &a[i]);
		b[i] = a[i];
	}
	sort(b + 1, b + 1 + n);
	tsize = unique(b + 1, b + 1 + n) - b - 1;
	t[0] = build(1, tsize);
	for (int i = 1; i <= n; i++) {
		t[i] = update(t[i - 1], find(a[i]));
	}
	int l, r, k;
	while (m--)
	{
		scanf("%d%d%d", &l, &r, &k);
		printf("%d\n", b[ask(t[l - 1], t[r], k)]);
	}
	return 0;
}

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值