在2016年,佳媛姐姐喜欢上了数字序列。因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题
,需要你来帮助他。这个难题是这样子的:给出一个1到n的全排列,现在对这个全排列序列进行m次局部排序,排
序分为两种:1:(0,l,r)表示将区间[l,r]的数字升序排序2:(1,l,r)表示将区间[l,r]的数字降序排序最后询问第q
位置上的数字。
标签:sum 顺序 tin gif getchar tor else while blog
输出数据仅有一行,一个整数,表示按照顺序将全部的部分排序结束后第q位置上的数字。
1 #include<cstdio> 2 #include<cstring> 3 #include<set> 4 const int N=1e5+10; 5 struct node{ 6 int l,r,id; 7 bool operator <(const node&p)const{return p.l>l;} 8 }; 9 typedef std::multiset<node>me; 10 typedef me::iterator IT; 11 int n,m,st[N*20],q,a[N],op,L,R,t=0,L0,R0; 12 me se; 13 struct point{int l,r,lson,rson,ok,sz;}e[25*N]; 14 int read(){ 15 int ans=0,f=1;char c=getchar(); 16 while(c<‘0‘||c>‘9‘){if(c==‘-‘)f=-1;c=getchar();} 17 while(c>=‘0‘&&c<=‘9‘){ans=ans*10+c-48;c=getchar();} 18 return ans*f; 19 } 20 void erase(int id){e[id].lson=e[id].rson=e[id].sz=e[id].ok=0;st[++t]=id;} 21 void build(int &rt,int l,int r,int x){ 22 rt=st[t--];e[rt].ok=0;e[rt].l=l;e[rt].r=r;e[rt].sz=1;e[rt].lson=e[rt].rson=0; 23 if(l==r)return; 24 int mid=(l+r)>>1; 25 if(mid>=x)build(e[rt].lson,l,mid,x); 26 else build(e[rt].rson,mid+1,r,x); 27 } 28 void spilt(int&rt,int last,int p,int x){ 29 rt=st[t--];e[rt].l=e[last].l;e[rt].r=e[last].r;e[rt].lson=e[rt].rson=0; 30 int ls=e[last].lson,rs=e[last].rson; 31 if(p){ 32 if(ls&&e[ls].sz>x)spilt(e[rt].lson,ls,p,x); 33 else { 34 e[rt].lson=ls;e[last].lson=0; 35 if(rs&&x!=e[ls].sz)spilt(e[rt].rson,rs,p,x-e[ls].sz); 36 } 37 } 38 else{ 39 if(ls&&e[ls].sz+1>=x){ 40 e[rt].rson=rs;e[last].rson=0; 41 if(x!=e[ls].sz+1)spilt(e[rt].lson,ls,p,x); 42 } 43 else if(rs)spilt(e[rt].rson,rs,p,x-e[ls].sz); 44 } 45 int ll=e[rt].lson,rr=e[rt].rson,ss=0; 46 if(ll)ss+=e[ll].sz;if(rr)ss+=e[rr].sz; 47 if(ss)e[rt].sz=ss,e[last].sz-=ss; 48 } 49 void merge(int rt,int rt0){ 50 int ll=0; 51 if(e[rt0].lson) 52 if(!e[rt].lson)e[rt].lson=e[rt0].lson; 53 else merge(e[rt].lson,e[rt0].lson); 54 if(e[rt0].rson) 55 if(!e[rt].rson)e[rt].rson=e[rt0].rson; 56 else merge(e[rt].rson,e[rt0].rson); 57 int l=e[rt].lson,r=e[rt].rson; 58 if(l)ll+=e[l].sz;if(r)ll+=e[r].sz; 59 e[rt].sz=ll; 60 erase(rt0); 61 } 62 int find(int rt,int k){ 63 if(e[rt].l==e[rt].r)return e[rt].l; 64 int l=e[rt].lson,r=e[rt].rson; 65 if(l&&e[l].sz>=k)return find(l,k); 66 else return find(r,k-e[l].sz); 67 } 68 int main(){ 69 n=read();m=read();int mid=(1+n)>>1; 70 for(int i=0;i<=19*N;i++)st[++t]=i; 71 for(int i=1;i<=n;i++){ 72 a[i]=read();int sum=st[t--]; 73 se.insert((node){i,i,sum});e[sum].l=1;e[sum].r=n;e[sum].ok=0;e[sum].lson=e[sum].rson=0;e[sum].sz=1; 74 if(a[i]<=mid)build(e[sum].lson,1,mid,a[i]); 75 else build(e[sum].rson,mid+1,n,a[i]); 76 } 77 while(m--){ 78 node pp;IT it1;bool flag=1; 79 op=read();L=read();R=read();L0=L;R0=R; 80 IT it=se.upper_bound((node){L,0,0}); 81 it--;int rt=it->id,rt0; 82 if(L==R)continue;if(L>R)std::swap(L,R); 83 if(L==it->l&&R==it->r){e[it->id].ok=op;continue;} 84 int p=0;if(e[it->id].ok)p=1,L=it->r-L+1;else L=L-it->l+1; 85 IT it0=se.upper_bound((node){R,0,0}); 86 it0--;rt0=it0->id; 87 if(e[it0->id].ok)R=it0->r-R+1;else R=R-it0->l+1; 88 if(L0>=it->l&&R0<=it->r){ 89 if(op==e[it->id].ok)continue; 90 L=L0-it->l;R=R0-it->l+2;if(e[it->id].ok)L=e[it->id].sz-L+1,R=e[it->id].sz-R+1; 91 if(L0!=it->l)spilt(rt,it->id,!p,L),e[rt].ok=!op,se.insert((node){it->l,L0-1,rt}); 92 if(L<R)R-=L; 93 if(R0!=it->r)spilt(rt0,it->id,p,R),e[rt0].ok=!op,se.insert((node){R0+1,it->r,rt0}); 94 node pp=*it;e[pp.id].ok=op;pp=(node){L0,R0,it->id}; 95 se.erase(it);se.insert(pp); 96 continue; 97 } 98 if(it->l<L0){ 99 spilt(rt,it->id,p,L); 100 pp=*it;pp.r=L0-1;se.erase(it); 101 it=se.insert(pp); 102 } 103 else flag=0;p=1;if(e[it0->id].ok)p=0; 104 if(it0->r>R0){ 105 spilt(rt0,it0->id,p,R);pp=*it0;it1=--it0;it0++;pp.l=R0+1;se.erase(it0);se.insert(pp);it0=it1; 106 } 107 else it1=it0,it0--,se.erase(it1); 108 merge(rt,rt0); 109 while(it0!=se.begin()&&it!=it0){ 110 it1=it0; 111 merge(rt,it1->id); 112 it0--;se.erase(it1); 113 } 114 e[rt].ok=op;if(!flag)se.erase(it); 115 se.insert((node){L0,R0,rt}); 116 117 } 118 q=read(); 119 IT it=se.upper_bound((node){q,0,0});it--; 120 q=q-it->l+1; 121 if(e[it->id].ok)q=e[it->id].sz-q+1; 122 printf("%d",find(it->id,q)); 123 return 0; 124 }
【bzoj4552/Tjoi2016&Heoi2016】排序——二分+线段树/平衡树+线段树分裂与合并
标签:sum 顺序 tin gif getchar tor else while blog
原文地址:http://www.cnblogs.com/JKAI/p/7639538.html