题解 P3372 【【模板】线段树 1】(珂朵莉树解法)

这道题可以用珂朵莉树做,但是由于数据比较不随机,而我也没有手写一颗平衡树,所以就被卡掉了,只拿了70分


 

珂朵莉树是一种基于平衡树的(伪)高效数据结构。

它的核心操作是推平一段区间。

简而言之,就是把之前的零零碎碎的都干掉,用一个美而饱满的大区间取代。

然后我们更新操作和查询操作就暴力遍历一遍,统计一下和就可以了。


 

Split操作

1 inline set<node>::iterator split(int pos){
2     set<node>::iterator it=s.lower_bound(node(pos));
3     if(it!=s.end()&&it->l==pos)return it;
4     --it;
5     int L=it->l,R=it->r;long long V=it->v;
6     s.erase(it),s.insert(node(L,pos-1,V));
7     return s.insert(node(pos,R,V)).first;
8 }

 

split操作就是获得区间的迭代器。

所以我们要先找到pos在哪里(也就是it)。

然后把它拆掉,再合在一起。

在这个过程中我们就可以拿到迭代器。(干什么等会说)


Update操作

1 inline void update(int l,int r,long long val=1){
2     set<node>::iterator itl=split(l),itr=split(r+1);
3     for(;itl!=itr;++itl)itl->v+=val;
4 }

 

现在刚才的split就派上用场了。

我们现在相当于是获得了区间的两端(也是区间)。

然后暴力遍历一下,给每个区间都打一个标记。


Query操作

1 inline long long query(int l,int r){
2     set<node>::iterator itl=split(l),itr=split(r+1);
3     long long ans=0;
4     for(;itl!=itr;++itl)ans+=(itl->r-itl->l+1)*itl->v;
5     return ans;
6 }

 

收集一下标记加在一起就可以了,思路和update一样。


 

70分代码如下:

3741ms 2368kb

  1 #include<bits/stdc++.h>
  2 
  3 using namespace std;
  4 
  5 namespace StandardIO{
  6 
  7     template<typename T>inline void read(T& x){
  8         x=0;T f=1;char c=getchar();
  9         for(;c<'0'||c>'9';c=getchar())if(c=='-')f=-1;
 10         for(;c>='0'&&c<='9';c=getchar())x=x*10+c-'0';
 11         x*=f;
 12     }
 13 
 14     template<typename T>inline void write(T x){
 15         if(x<0)putchar('-'),x*=-1;
 16         if(x>=10)write(x/10);
 17         putchar(x%10+'0');
 18     }
 19 
 20 }
 21 
 22 using namespace StandardIO;
 23 
 24 namespace ChthollyTree{
 25     
 26     struct Tree{
 27         private:
 28             struct node{
 29                 int l,r;mutable long long v;
 30                 node(int L,int R=-1,long long V=0):l(L),r(R),v(V){}
 31                 bool operator < (const node &o)const{
 32                     return l<o.l;
 33                 } 
 34             };
 35             set<node>s;
 36             
 37             inline set<node>::iterator split(int pos){
 38                 set<node>::iterator it=s.lower_bound(node(pos));
 39                 if(it!=s.end()&&it->l==pos)return it;
 40                 --it;
 41                 int L=it->l,R=it->r;long long V=it->v;
 42                 s.erase(it),s.insert(node(L,pos-1,V));
 43                 return s.insert(node(pos,R,V)).first;
 44             }
 45             
 46             inline void update(int l,int r,long long val=1){
 47                 set<node>::iterator itl=split(l),itr=split(r+1);
 48                 for(;itl!=itr;++itl)itl->v+=val;
 49             }
 50             
 51             inline long long query(int l,int r){
 52                 set<node>::iterator itl=split(l),itr=split(r+1);
 53                 long long ans=0;
 54                 for(;itl!=itr;++itl)ans+=(itl->r-itl->l+1)*itl->v;
 55                 return ans;
 56             }
 57             
 58         public:
 59             Tree(){}
 60             ~Tree(){}
 61             
 62             inline void Init(int n){
 63                 s.insert(node(1,n));
 64             }
 65             
 66             inline void Update(int l,int r,long long val){
 67                 update(l,r,val);
 68             }
 69             
 70             inline int Query(int l,int r){
 71                 return query(l,r); 
 72             }
 73     };
 74     
 75 }
 76 
 77 using namespace ChthollyTree;
 78 
 79 namespace Solve{
 80     
 81     const int N=100100;
 82     
 83     int n,m;
 84     int sum[N];
 85     Tree ljz;
 86     
 87     inline void solve(){
 88         read(n),read(m);
 89         ljz.Init(n);
 90         for(register int i=1;i<=n;++i){
 91             int tmp;read(tmp);
 92             sum[i]=sum[i-1]+tmp;
 93         } 
 94         while(m--){
 95             int op,x,y,z;
 96             read(op),read(x),read(y);
 97             if(op==1){
 98                 read(z);
 99                 ljz.Update(x,y,z);
100             }else{
101                 write(ljz.Query(x,y)+sum[y]-sum[x-1]),putchar('\n');
102             }
103         }
104     }
105 
106 }
107 
108 using namespace Solve;
109 
110 int main(){
111     solve();
112 }

 

posted @ 2018-10-24 21:37  Ilverene  阅读(586)  评论(1编辑  收藏  举报