主席树设计与实现

Posted rikka

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了主席树设计与实现相关的知识,希望对你有一定的参考价值。

主席树设计与实现

一、主席树的一句话简介

  1、主席树是可持久化线段树

  2、可持久化技术用于将多棵树相同的部分复用、不同的部分分开构成一颗新树。

  3、主席树的实际物理原型是多颗线段树。

 

二、区间第K大问题

  1、设计上应当考虑,如果有了多颗线段树,应当怎么做这道题?

  2、有了思路之后应当考虑,可持久化线段树的性质是什么样的,使得我们应当如何针对这种特殊的性质完成线段树的设计?

  3、如何进行更新操作?

 

ans:

  1、如果有多颗线段树,则每次插入新的元素建立一颗的线段树;在查找的时候,可以知道对于任意元素,有多少个元素小于等于给定元素。

  2、想象可得,每个点的更新都将造成从最底部到根节点的一系列改变。因此,可得:根节点必然每次改变,且新的更节点可以带出新的其余替换节点。

  3、考虑使用上一次的根节点的若干节点,在一路向下的过程中不断进行节点复制。这种写法可以相对优雅的完成主席树的更新。

 

题目:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1175

技术分享图片
  1 #include<iostream>
  2 #include<algorithm>
  3 using namespace std;
  4 
  5 #define ll long long 
  6 #define idx(x) (lower_bound(mapp,mapp+n,x)-mapp)
  7 
  8 
  9 const ll MAXN=1<<20;
 10 
 11 int head[50233];
 12 int n,m;
 13 class Node
 14 {
 15     public:
 16         int l,r,number,lc,rc;
 17 
 18 };Node tree[MAXN];
 19 int tree_cnt ;
 20 int mapp[50233];
 21 int arr[50233];
 22 
 23 void build(int a,int b,int now)
 24 {
 25     tree_cnt ++ ;
 26     tree[now].l = a;
 27     tree[now].r = b;
 28     tree[now].number = 0;
 29     if(a == b-1)return ;
 30     int mid = (a+b)/2;
 31     tree[now].lc = tree_cnt;
 32     build(a,mid,tree_cnt);
 33     tree[now].rc = tree_cnt;
 34     build(mid,b,tree_cnt);
 35 }
 36 
 37 void insert(int pos,int key,int times,int old)
 38 {
 39         
 40     int now = tree_cnt++;
 41     int l = tree[old].l;
 42     int r = tree[old].r;
 43 
 44     tree[now].l = l;
 45     tree[now].r = r;
 46     tree[now].number = tree[old].number;
 47     if(l == r-1)
 48     {
 49         tree[now].number += key;
 50         return ;
 51     }
 52     int mid = (l+r)/2;
 53     
 54     tree[now].lc = tree[old].lc;
 55     tree[now].rc = tree[old].rc;
 56     if(pos < mid)
 57     {
 58         tree[now].lc = tree_cnt;
 59         insert(pos,key,times,tree[old].lc);
 60     }else{
 61         tree[now].rc = tree_cnt;
 62         insert(pos,key,times,tree[old].rc);
 63     }
 64     tree[now].number = tree[tree[now].lc].number + tree[tree[now].rc].number;
 65     return ;
 66 }
 67 
 68 int search(int pos,int now)
 69 {
 70     int l = tree[now].l;
 71     int r = tree[now].r;
 72     if(l == r-1)return tree[now].number;
 73     int mid = (l+r)/2;
 74     int lc = tree[now].lc;
 75     int rc = tree[now].rc;
 76     if(mid <= pos)return tree[lc].number+search(pos,rc);
 77     else return search(pos,lc);
 78 }
 79 
 80 int check(int a,int b,int num)
 81 {
 82     int cntt = search(num,head[b+1]);
 83     cntt -= search(num,head[a]);
 84     return cntt;
 85 }
 86 
 87 int bin_search(int a,int b,int l,int r,int num)
 88 {
 89     if(l == r-1)return mapp[r];
 90     int mid = (l+r)/2;
 91     int cntt = check(a,b,mid);
 92     if(cntt<num)return bin_search(a,b,mid,r,num);
 93     else return bin_search(a,b,l,mid,num);
 94 }
 95 
 96 void init()
 97 {
 98     tree_cnt = 1;
 99     head[0] = 1;
100     build(0,n,1);
101     for(int i=0;i<n;++i)
102     {
103         cin>>arr[i];
104         mapp[i] = arr[i];
105     }sort(mapp,mapp+n);
106     cin>>m;
107     for(int i=0;i<n;++i)
108     {
109         head[i+1] = tree_cnt;
110         insert(idx(arr[i]),1,i+1,head[i]);
111     }
112     for(int i=0;i<m;++i)
113     {
114         int a,b,c;
115         cin>>a>>b>>c;
116         cout<<bin_search(a,b,-1,n,b-a+2-c)<<"
";
117     }
118 }
119 int main()
120 {
121     cin.sync_with_stdio(false);
122     while(cin>>n)init();
123     // build(0,233,1);
124 
125     return 0;
126 }
View Code

 

以上是关于主席树设计与实现的主要内容,如果未能解决你的问题,请参考以下文章

2020-2021年度第二届全国大学生算法设计与编程挑战赛 (春季赛)- 天才的操作(线段树+主席树+树上倍增)

可持久化线段树(主席树)

模板主席树

可持久化专题——浅谈主席树:可持久化线段树

主席树

主席树