P2617 Dynamic Rankings(主席树+树状数组)

Posted zhangbuang

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了P2617 Dynamic Rankings(主席树+树状数组)相关的知识,希望对你有一定的参考价值。

怕是还没有题解,所以先写一篇。

这题就是维护带修改的主席树。首先树套树肯定是能做的,既然树套树能做那么整体二分肯定也是可以的。

由于我并没有使用这两种做法,所以此处不予介绍。

大概描述下主席树的思路:

首先说说怎么搞带修改主席树?

回忆一般的kth问题,我们的主席树求的是前缀和,这样我们在目标区间的左右端点的主席树差分下就能求出kth。

那么我们如何支持修改操作?

考虑到我们之前使用主席树朴素的维护区间前缀和,支持修改的话,只要把前缀和交给擅长它的树状数组维护,主席树只要维护下位置就好。


 


 


 

 

 1 #include"bits/stdc++.h"
 2 using namespace std;
 3 #define nn 40000000
 4 #define lowbit(x) ((x)&(-x))
 5 
 6 int rt[nn],b[nn],a[nn],ca[nn],cb[nn],cc[nn] ;  
 7 int size[nn],xx[nn],yy[nn],l[nn],r[nn];
 8 int n,m;
 9 int totn;
10 int tot,totx,toty;
11 
12 inline int read(){
13     int f=1,x=0;char ch;
14     do{ch=getchar();if(ch==-)f=-1;}while(ch<0||ch>9);
15     do{x=x*10+ch-0;ch=getchar();}while(ch>=0&&ch<=9);
16     return f*x;
17 }
18 
19 
20 void change(int &y,int x,int L,int R,int pos,int v)
21 {
22       y=++tot; size[tot]=size[x]+v,l[tot]=l[x],r[tot]=r[x];
23     if (L==R)return ;
24     int mid = R+L>>1;if (pos<=mid)change(l[tot],l[x],L,mid,pos,v);
25     else change(r[tot],r[x],mid+1,R,pos,v);
26 }
27 
28 inline void add(int x,int v)
29 {
30     int k = lower_bound(b+1,b+1+totn,a[x])-b;
31     for (int i=x;i<=n;i+=lowbit(i)) change(rt[i],rt[i],1,totn,k,v);
32 }
33 
34 int query(int L,int R,int p)
35 {
36     if (L==R)return R; int sum=0; int mid = L+R>>1;
37     for (int i=1;i<=totx;i++)sum-=size[l[xx[i]]];
38     for (int i=1;i<=toty;i++)sum+=size[l[yy[i]]];
39     if (p<=sum)
40     {
41        for ( int i=1;i<=totx;i++)xx[i]=l[xx[i]];
42        for ( int i=1;i<=toty;i++)yy[i]=l[yy[i]];
43        return query(L,mid,p);
44     }
45      for ( int i=1;i<=totx;i++)xx[i]=r[xx[i]];
46        for ( int i=1;i<=toty;i++)yy[i]=r[yy[i]];
47       return query(mid+1,R,p-sum);
48 }
49 
50 int main()
51 {  char s;
52     n=read(),m=read();
53     for (int i=1;i<=n;i++)a[i]=read(),b[++totn]=a[i];
54 
55     for (int i=1;i<=m;i++) 
56     {
57         cin>>s; if (s==Q) ca[i]=read(),cb[i]=read(),cc[i]=read();
58         else ca[i]=read(),cb[i]=read(),b[++totn]=cb[i];
59     }
60         sort(b+1,b+1+totn);
61     totn=unique(b+1,b+1+totn)-b-1;
62     
63     for ( int i=1;i<=n;i++) add(i,1); 
64     for ( int i=1;i<=m;i++)
65     {
66         if (cc[i]) 
67         {  totx=toty=0;
68             for ( int j=ca[i]-1;j;j-=lowbit(j))xx[++totx]=rt[j];
69             for ( int j=cb[i];j;j-=lowbit(j))yy[++toty]=rt[j];
70             printf("%d
",b[query(1,totn,cc[i])]);
71         }
72         else 
73         {
74             add(ca[i],-1),a[ca[i]]=cb[i],add(ca[i],1);
75         }
76     }
77     
78 }

 

以上是关于P2617 Dynamic Rankings(主席树+树状数组)的主要内容,如果未能解决你的问题,请参考以下文章

「Luogu P2617」Dynamic Rankings

P2617 Dynamic Rankings

bzoj P2617 Dynamic Rankings

P2617 Dynamic Rankings (动态开点权值线段树 + 树状数组)

洛谷 P2617 Dynamic Rankings

ybt金牌导航4-6-6luogu P2617动态排名 / Dynamic Rankings