主席树 从入门到入门

前言:

树剖板子早就打腻了。一直想学习一些新的模板,今天就学习一下主席树。
其实是因为暑假集训的东西我全忘了

1. luogu P3834 【模板】可持久化线段树 2(主席树)

最板子的一道题,然而看了好长时间才看明白,震惊,我竟然折磨菜,活到爆!
主席树经典用处:求静态区间第k大。
先对a数组离散化,然后对a的每一个前缀开一个权值线段树,查询[l,r]的第k大时,需要线段树上二分,分别找到[1,r]中有几个大于mid,[1,l-1]中有多少大于mid,然后相减,就是[l,r]中大于mid的个数。
代码:

#include <bits/stdc++.h>
using namespace std;
const int maxn=200000+10;
#define gc() (p1 == p2 ? (p2 = buf + fread(p1 = buf, 1, 1 << 20, stdin), p1 == p2 ? EOF : *p1++) : *p1++)
#define read() ({ register int x = 0, f = 1; register char c = gc(); while(c < '0' || c > '9') { if (c == '-') f = -1; c = gc();} while(c >= '0' && c <= '9') x = x * 10 + (c & 15), c = gc(); f * x; })
char buf[1 << 20], *p1, *p2;
int a[maxn],b[maxn],root[maxn<<5];
struct node{
	int ls,rs,size;
}tree[maxn<<5];
int n,m,tot;
bool cmp(int x,int y){
	return x<y;
}
void update(int &u,int l,int r,int x){
	int v=++tot;
	tree[v]=tree[u];
	u=v;
	tree[v].size++;
	if(l==r) return;
	int mid=(l+r)>>1;
	if(x<=mid) update(tree[u].ls,l,mid,x);
	else update(tree[u].rs,mid+1,r,x);
}
int query(int s,int t,int k){
	s=root[s];
	t=root[t];
	int mid,sum,l=1,r=b[0];
	while(l<r){
		mid=(l+r)>>1;
		sum=tree[tree[t].ls].size-tree[tree[s].ls].size;
		if(sum>=k){
			s=tree[s].ls;
			t=tree[t].ls;
			r=mid;
		}else{
			k-=sum;
			s=tree[s].rs;
			t=tree[t].rs;
			l=mid+1;
		}
	}
	return l;
}
void Solve(){
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;++i){
		scanf("%d",&a[i]);
		b[i]=a[i];
	}
	sort(b+1,b+n+1,cmp);
	b[0]=unique(b+1,b+n+1)-b-1;
	root[0]=++tot;
	for(int i=1;i<=n;++i){
		root[i]=root[i-1];
		int x=lower_bound(b+1,b+b[0]+1,a[i])-b;
		update(root[i],1,b[0],x);
	}
	int l,r,k;
	while(m--){
		scanf("%d%d%d",&l,&r,&k);
		printf("%d\n",b[query(l-1,r,k)]);
	}
}
int main(){
	Solve();
	return 0;
}


这里我在纠结一个问题,就是二分的过程。
对于普通的二分答案,我一般是这么写的:

??

while(l<=r){
	int mid=(l+r)>>1;
	if(check(mid)) r=mid-1;
	else l=mid+1;
}
return l;

但是搬到这里,他就不对了。

int query(int s,int t,int k){//这是错误代码
	s=root[s];
	t=root[t];
	int mid,sum,l=1,r=b[0];
	while(l<=r){
		mid=(l+r)>>1;
		sum=tree[tree[t].ls].size-tree[tree[s].ls].size;
		if(sum>=k){
			s=tree[s].ls;
			t=tree[t].ls;
			r=mid-1;
		}else{
			k-=sum;
			s=tree[s].rs;
			t=tree[t].rs;
			l=mid+1;
		}
	}
	return l;
}

反正我就挺迷惑。
后来张大仙给出了一个令我信服的解释。
其实这压根不是二分答案,这个while循环模拟的是线段树上二分。
换成递归版本应该是这样:

inline int query(int u, int v, int l, int r, int k)//随便扒了一份题解上的代码
{
    if (l >= r) return l;
    int x = sum[L[v]] - sum[L[u]];
    if (x >= k) return query(L[u], L[v], l, mid, k);
    else return query(R[u], R[v], mid+1, r, k-x);
}

现在就很清晰了。

posted @ 2020-10-22 15:45  “起个名字真难♘”  阅读(77)  评论(2编辑  收藏  举报