初识Splay树

yyb大佬的入门讲解:http://www.cnblogs.com/cjyyb/p/7499020.html

Orz

我就补充一下Splay树怎么作为区间树使用:

1、Splay树作为一种二叉排序树,对其中序dfs遍历后的结果便是按key值从小到大排序的数列,那么我们将编号作为key值即可。

2、如果要获取到[l,r]区间的数列,则利用Splay操作将l-1作为根,r+1作为根的右节点,再对r+1的左子树中序遍历。

3、上面似乎并没有体现区间树的效率优势。。

4、模仿线段树的做法,每个节点除了维护自身的信息外,还需维护以自己为根的这棵树对应的区间的信息,维护的信息会在rotate操作时发生改变,则在rotate操作时更新节点的信息。

5、lazy_tag,在节点信息变更或查询时,记得提前把lazy_tag下传。

洛谷的区间翻转裸题:

  1 #include<iostream>
  2 #include<cstdio>
  3 #include<cstdlib>
  4 #include<cstring>
  5 #include<cmath>
  6 #include<algorithm>
  7 #include<map>
  8 using namespace std;
  9 #define N 100005
 10 struct Node{
 11     int ch[2]; //左右儿子
 12     int val;
 13     int ff;
 14     int size;
 15     int tag; 
 16 }node[N];
 17 int tot=1,root=0;
 18 inline void push_up(int x)  //refresh
 19 {    int lson=node[x].ch[0];
 20     int rson=node[x].ch[1];
 21     node[x].size=node[lson].size+node[rson].size+1;
 22 }
 23 inline void push_down(int x) //down lazy_tag
 24 {    if(node[x].tag)
 25     {
 26     node[x].tag=0;
 27     int lson=node[x].ch[0];
 28     int rson=node[x].ch[1];
 29     node[lson].tag^=1;
 30     node[rson].tag^=1;
 31     swap(node[x].ch[0],node[x].ch[1]);
 32     }
 33 }
 34 inline void rotate1(int x)
 35 {
 36     int y=node[x].ff;
 37     int z=node[y].ff;
 38     int k=(node[y].ch[1]==x);
 39     node[z].ch[node[z].ch[1]==y]=x;
 40     node[x].ff=z;
 41     
 42     node[y].ch[k]=node[x].ch[k^1];
 43     node[node[x].ch[k^1]].ff=y;
 44     
 45     node[x].ch[k^1]=y;
 46     node[y].ff=x;
 47     push_up(y);
 48     push_up(x);
 49 }
 50 inline void splay(int x,int goal) //把x节点伸展到goal节点下 
 51 {
 52     while(node[x].ff!=goal)
 53     {
 54         int y=node[x].ff;
 55         int z=node[y].ff;
 56         if(z!=goal)
 57             (node[y].ch[0]==x)^(node[z].ch[0]==y)?rotate1(x):rotate1(y);
 58         rotate1(x);
 59     }
 60     if(goal==0)root=x;
 61 }
 62 inline void ins(int v)
 63 {
 64     int u=root,ff=0;
 65     while(u&&node[u].val!=v)
 66     {    ff=u;
 67         u=node[u].ch[v>node[u].val];
 68     }
 69     if(u) ;
 70     else
 71     {
 72         u=tot++;
 73         node[ff].ch[v>node[ff].val]=u;
 74         node[u].ff=ff;
 75         node[u].ch[0]=node[u].ch[1]=0;
 76         node[u].val=v;
 77         node[u].size=1;
 78     }
 79     splay(u,0);
 80 }
 81 inline int find_k(int k)
 82 {    int u=root;
 83     while(u)
 84     {    push_down(u);
 85         int lson=node[u].ch[0];
 86         if(node[lson].size==k-1) return u;
 87         if(node[lson].size<k-1)
 88         {
 89             u=node[u].ch[1];
 90             k-=node[lson].size+1;
 91         }
 92         else
 93         {
 94             u=node[u].ch[0];
 95         }
 96     }
 97     return -1;
 98 }
 99 int n,m,a[N],tt=0;
100 void dfs_w(int x)
101 {    if(!x) return;
102     push_down(x);
103     dfs_w(node[x].ch[0]);
104     a[tt++]=node[x].val;
105     dfs_w(node[x].ch[1]);    
106 }
107 int main()
108 {
109     
110     cin>>n>>m;
111     for(int i=1;i<=n+2;i++)  //增加头尾两个哨兵 
112     {
113         ins(i);
114     }
115     while(m--)
116     {
117         int l,r;
118         scanf("%d%d",&l,&r);
119         l=find_k(l);
120         r=find_k(r+2);
121         splay(l,0);
122         splay(r,l);
123         node[node[r].ch[0]].tag^=1;
124     }
125     dfs_w(root);
126     for(int i=1;i<n;i++)
127         printf("%d ",a[i]-1);
128     printf("%d\n",a[n]-1);
129     
130     return 0;
131 }

 

posted @ 2018-07-13 12:53  hzhuan  阅读(131)  评论(0编辑  收藏  举报