青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

算法學(xué)社
記錄難忘的征途
posts - 141,comments - 220,trackbacks - 0

題目描述:

   對(duì)一個(gè)字符串S(初始為空),有Q次操作(Q<=50,000),操作分三種:
      1. 在某個(gè)位置p后面插入一個(gè)長(zhǎng)度不大于100的字符串。
      2. 刪除一段字符[l,r]
      3. 輸出在第k次操作時(shí),字符串(S_l ... S_r) 插入的字符不超過(guò)1,000,000個(gè)。

吐槽:

memory limit真是極限

算法分析:

支持任意區(qū)間的插入刪除,我們比較熟悉的是splay tree。可惜splay無(wú)法可持久化(至于為啥不知道 ><....)。
其次是塊鏈,空間和時(shí)間復(fù)雜度都是Q*sqrt(N),難以接受。
其實(shí)支持這種操作的還有就是treap了,也是我唯一知道的可以可持久化的balanced tree。(AVL什么的也可以的哦,fanhq巨巨的blog有寫(xiě),可惜我不會(huì))

treap可以在O(logn)的時(shí)間內(nèi)split和merge,具體操作CLJ論文有寫(xiě)。正好可以和插入一段/刪除一段對(duì)上,而且連rotate都不需要了。
這兩個(gè)操作可以保證heap的性質(zhì),也就同時(shí)能保證期望高度了。

注意逐個(gè)插入字符的時(shí)候還是要非持久化,否則空間是N*loglen + Q*logN的。而且要手動(dòng)分配連續(xù)空間,否則都會(huì)MLE。
  1 #include<iostream>
  2 #include<cstdlib>
  3 #include<cstdio>
  4 #include<cstring>
  5 #include<cassert>
  6 using namespace std;
  7 #define obfused
  8 // treap
  9 int flag = 0,ccnt = 0, memtop = 0;
 10 struct node {
 11     int key,sz,val;
 12     char ch;
 13     node *lc, *rc;
 14     node(){lc = rc = NULL;}
 15     void sumup(){
 16             if(lc == NULL && rc == NULL) sz = 1;
 17             else if(lc == NULL) sz = rc -> sz + 1;
 18             else if(rc == NULL) sz = lc -> sz + 1;
 19             else sz = rc -> sz + lc -> sz + 1;
 20     }
 21     node(char _c,int _key,int _val, node* _l,node *_r):
 22         val(_val) ,lc(_l), rc(_r),ch(_c),key(_key){
 23             ccnt++;
 24             sumup();
 25         }
 26     node* mymerge(node*);
 27     node* merge(node*);
 28     void output(int);
 29     pair<node*,node*> split(int);
 30     void op();
 31     void opc();
 32 };
 33 // memory pool
 34 node mem[3000005];
 35 node* mknode(char c,int key,int val,node* L,node *R){
 36     mem[memtop] = node(c,key,val,L,R);
 37     memtop++;
 38     return &mem[memtop-1];
 39 }
 40 // build
 41 const int N = 50005;
 42 node *root[N];
 43 char ch[105];
 44 node* make(char *ch,int pos){
 45     int n = strlen(ch);
 46     node * ans = mknode(ch[0],rand(),pos,NULL,NULL);
 47     for(int i = 1; i < n; i++){
 48         node *tmp = mknode(ch[i],rand(),pos+i,NULL,NULL);
 49         ans = ans -> mymerge(tmp);
 50     }
 51     return ans;
 52 }
 53 // treap's member function
 54 node* node::mymerge(node *t){
 55         if(t==NULL) return this;
 56         if(key >= t->key){
 57             if(rc == NULL) 
 58                 rc = t;
 59             else rc = rc -> mymerge(t);
 60             sumup();
 61             return this;
 62         } else {
 63             t -> lc = this;
 64             t -> sumup();
 65             return t;
 66         }
 67     }
 68 node* node::merge(node *t){
 69         if(t == NULL) return this;
 70         if(key >= t-> key) {
 71             if(rc == NULL) {
 72                 node *nd = mknode(ch,key,val,lc,t);
 73                 return nd;
 74             } else {
 75                 node *nd = mknode(ch,key,val,lc,rc->merge(t));
 76             }
 77         } else {
 78             if(t -> lc == NULL){
 79                 node *nd = mknode(t->ch,t->key,t->val,this,t->rc);
 80                 return nd;
 81             } else {
 82                 node *nd = mknode(t->ch,t->key,t->val,merge(t->lc),t->rc);
 83             }
 84         }
 85     }
 86 pair<node*,node*> node:: split(int len){
 87 //        cout<<"split: "<<ch<<" "<<len<<endl;
 88         pair<node*,node*> pr;
 89         int cnt = 0;
 90         if(lc != NULL) cnt = lc -> sz;
 91         if(cnt >= len) {
 92             if(lc == NULL) return make_pair((node*)NULL,this);
 93             else {
 94                 pr = lc -> split(len);
 95                 node *L = pr.first;
 96                 node *R = mknode(ch,key,val,pr.second,rc);
 97                 return make_pair(L,R);
 98             }
 99         } else {
100             if(rc == NULL) return make_pair(this,(node*)NULL);
101             else {
102                 pr = rc -> split(len -cnt - 1);
103                 node *L = mknode(ch,key,val,lc,pr.first);
104                 node *R = pr.second;
105                 return make_pair(L,R);
106             }
107         }
108     }
109 void node:: output(int k){
110         int cnt = 0;
111         assert(k >=1 && k <= sz);
112         if(lc != NULL) cnt = lc -> sz;
113         if(k <= cnt) lc -> output(k);
114         else if(k == cnt + 1) {
115             printf("%c",ch);
116             if(ch == 'c') flag ++;
117         } else {
118             assert(rc != NULL);
119             rc -> output(k - cnt - 1);
120         }
121     }
122 void node:: op(){
123         cout<< ch <<" "<<val<<" "<<sz<<endl;
124         if(lc != NULL) cout<<"lc: "<<lc->ch<<endl;
125         if(rc != NULL) cout<<"rc: "<<rc->ch<<endl;
126         if(lc != NULL) lc->op();
127         if(rc != NULL) rc->op();
128     }
129 void node:: opc(){
130         if(lc != NULL) lc->opc();
131         cout<<ch;
132         if(rc != NULL) rc->opc();
133     }
134 // main && io
135 int main(){
136     int q,count = 0;
137     cin >> q;
138     for(int _=0;_<q;_++){
139         //cout<<"now: "<<_<<endl;
140         int cmd,pos,len,ver;
141         scanf("%d",&cmd);
142         if(cmd == 1) {
143             scanf("%d%s",&pos,ch);
144             #ifdef obfused
145             pos -= flag;
146             #endif
147             //cout<<pos<<" "<<ch<<endl;
148             node *t = make(ch,pos+1);
149             if(root[count] == NULL) {
150                 root[++count] = t;
151             } else {
152                 pair<node*,node*>pr = root[count]->split(pos);
153                 t = t->merge(pr.second);
154                 if(pr.first != NULL) t = pr.first->merge(t);
155                 root[++count] = t;
156             }
157         } else if(cmd == 2){
158             scanf("%d%d",&pos,&len);
159             #ifdef obfused
160             pos -= flag,len -= flag;
161             #endif
162 //            cout<<pos<<" "<<len<<endl;
163             pair<node*,node*>pr = root[count]->split(pos-1);
164             node* u = pr.first;
165 //            if(u!=NULL){cout<<"u: "; u->op();u->opc(); cout<<endl;}
166             assert(pr.second != NULL);
167             pair<node*,node*>prn = pr.second-> split(len);
168             node* v = prn.second;
169             node* ans;
170             if(u == NULL) ans = v;
171             else ans = u->merge(v);
172 //            if(v!=NULL){cout<<"v: "; v->op();v->opc(); cout<<endl;}
173 //            ans -> opc(); cout<<endl;
174             root[++count] = ans;
175 //            cout<<"count: "<<count<<endl;
176         } else {
177             scanf("%d%d%d",&ver,&pos,&len);
178             #ifdef obfused
179             pos -= flag, ver -= flag, len -= flag;
180             #endif
181 //            cout<<ver<<" "<<pos<<" "<<len<<endl;
182             for(int i = 0; i < len; i++)
183                 root[ver] -> output(pos + i);
184             printf("\n");
185         }
186         //root[count]->op();
187 //        root[count]->opc();cout<<endl;
188 //        cout<<ccnt<<endl;
189     }
190 }
191 
posted on 2013-03-19 22:15 西月弦 閱讀(1600) 評(píng)論(1)  編輯 收藏 引用 所屬分類(lèi): 解題報(bào)告

FeedBack:
# re: uva 12583 可持久化treap
2013-03-24 09:58 | wuyiqi
好久沒(méi)看你寫(xiě)題解,一來(lái)就是這么神的題。。。。。  回復(fù)  更多評(píng)論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            伊人夜夜躁av伊人久久| 欧美一区二区在线视频| 性色av一区二区怡红| 亚洲美女性视频| 亚洲精品你懂的| 日韩视频中午一区| 日韩视频在线观看一区二区| 欧美中文在线免费| 欧美亚洲日本网站| 久久久久久久成人| 亚洲视频在线观看免费| 欧美激情国产日韩精品一区18| 在线不卡视频| 久久精品夜夜夜夜久久| 欧美激情成人在线| 国产精品二区三区四区| 国内精品久久久久国产盗摄免费观看完整版 | 亚洲黄色免费网站| 99国产精品久久久久老师| 亚洲视屏一区| 免费亚洲一区二区| 亚洲神马久久| 老司机午夜精品视频在线观看| 欧美精品一区二区精品网| 国产欧美激情| 亚洲精品中文字幕在线| 亚洲欧美卡通另类91av| 欧美成人午夜| 亚洲一区在线直播| 欧美久久久久久久久久| 国产一区二区三区奇米久涩| 亚洲理伦在线| 裸体女人亚洲精品一区| 一区二区三区高清在线 | 欧美日韩在线观看视频| 国产一区视频观看| 亚洲影视综合| 亚洲国产精品日韩| 久久国产精品色婷婷| 国产精品九九| 国内精品久久久久久| 久久精品国产亚洲5555| 免费成人性网站| 亚洲一区制服诱惑| 久久九九国产精品怡红院| 亚洲精品国产精品国自产观看浪潮 | 亚洲国产你懂的| 日韩视频免费观看高清在线视频| 国产精品久久久久久久久久免费 | 老司机精品久久| 老司机精品视频一区二区三区| 日韩亚洲一区二区| 亚洲一区区二区| 亚洲第一精品夜夜躁人人爽| 亚洲日本欧美| 国产午夜精品在线| 亚洲精品一二区| 狠狠色狠狠色综合人人| aa级大片欧美三级| 亚洲成人原创| 午夜精彩视频在线观看不卡| 亚洲美女在线观看| 欧美一区日韩一区| 亚洲网在线观看| 男人插女人欧美| 久久久久国产精品人| 欧美三级欧美一级| 亚洲国产黄色| 国产亚洲成av人在线观看导航| 亚洲精品乱码久久久久久按摩观| 韩国三级在线一区| 亚洲午夜国产一区99re久久| 亚洲精品久久久久久久久| 久久精品日产第一区二区| 午夜久久久久久久久久一区二区| 欧美精品一区二区精品网| 欧美成人午夜影院| 国模套图日韩精品一区二区| 亚洲一二三区精品| 亚洲小说欧美另类婷婷| 欧美日本韩国| 亚洲精品久久久久久久久久久| 黄色成人av网| 欧美亚洲免费在线| 午夜精彩视频在线观看不卡| 欧美极品在线观看| 亚洲国产人成综合网站| 在线播放一区| 久久精品九九| 久久久久久噜噜噜久久久精品| 国产精品视频yy9099| 亚洲亚洲精品三区日韩精品在线视频| 正在播放欧美视频| 欧美日韩一区二区三区在线看| 91久久香蕉国产日韩欧美9色| 亚洲人成欧美中文字幕| 欧美电影免费观看高清| 亚洲缚视频在线观看| 亚洲人成高清| 欧美日韩影院| 在线一区亚洲| 欧美一级午夜免费电影| 国产欧美91| 欧美自拍偷拍| 免费人成网站在线观看欧美高清| 亚洲成在线观看| 欧美成人在线免费观看| 亚洲人成小说网站色在线| 亚洲精品美女久久7777777| 欧美激情一二三区| 亚洲一区二区三区免费在线观看| 欧美一级二级三级蜜桃| 激情五月综合色婷婷一区二区| 久久亚洲影院| 亚洲精品裸体| 免费日韩成人| 夜夜嗨av一区二区三区四区| 精品69视频一区二区三区| 夜夜嗨av色一区二区不卡| 黄网站色欧美视频| 亚洲午夜精品久久久久久浪潮| 亚洲高清在线| 欧美在线观看天堂一区二区三区| 亚洲综合日本| 欧美日韩另类丝袜其他| 欧美黄在线观看| 国产视频在线观看一区| 一本大道av伊人久久综合| 亚洲日韩中文字幕在线播放| 久久久久国产一区二区三区四区| 久久精品最新地址| 国产欧美精品一区二区色综合 | 在线播放豆国产99亚洲| 欧美一区免费| 久久久久国产一区二区三区| 国产午夜精品久久久久久免费视| 亚洲影视在线播放| 亚洲欧美日韩一区二区三区在线观看 | 久久精品99国产精品酒店日本| 国产精品国产亚洲精品看不卡15| 亚洲精品视频在线观看网站 | 欧美jizz19hd性欧美| 国模私拍一区二区三区| 久久久999精品免费| 卡一卡二国产精品| 在线观看91精品国产麻豆| 麻豆精品精华液| 亚洲国产精品久久久久秋霞蜜臀| 亚洲国产另类精品专区 | 亚洲麻豆av| 在线视频欧美精品| 欧美午夜片在线免费观看| 亚洲视频在线免费观看| 欧美综合77777色婷婷| 国内精品久久久久久久果冻传媒 | 国产欧美日韩综合一区在线播放 | 久久久久久久久久久成人| 亚洲综合精品四区| 国产精品你懂的在线欣赏| 午夜激情综合网| 久久国产视频网站| 一色屋精品视频在线观看网站| 裸体素人女欧美日韩| 日韩一区二区高清| 香蕉久久夜色精品国产使用方法 | 亚洲午夜伦理| 欧美在线观看你懂的| 国产自产2019最新不卡| 国产欧美婷婷中文| 国产精品乱码一区二区三区| 欧美美女bbbb| 亚洲一区三区视频在线观看| 亚洲美女色禁图| 国产亚洲精品自拍| 欧美精品久久99| 亚洲主播在线| 亚洲国产va精品久久久不卡综合| 亚洲尤物精选| 亚洲人成亚洲人成在线观看| 国产精品久久久久免费a∨| 久久深夜福利免费观看| 一区二区三区四区五区精品视频| 久久裸体艺术| 亚洲综合精品自拍| 亚洲精选在线| 极品av少妇一区二区| 欧美性做爰毛片| 欧美大片网址| 久久国产毛片| 午夜国产精品影院在线观看| 亚洲人成77777在线观看网| 久久久精品网| 亚洲自拍啪啪| 99精品国产一区二区青青牛奶| 一区二区亚洲精品国产| 国产乱码精品一区二区三区五月婷 | 欧美在线观看一区二区三区| 亚洲欧洲综合| 亚洲成人在线视频网站| 国产一区二区三区四区五区美女 |