给定一个含有n个数的序列a[1],a[2],a[3]……a[n],程序必须回答这样的询问:对于给定的i,j,k,在a[i],a[i+1
],a[i+2]……a[j]中第k小的数是多少(1≤k≤j-i+1),并且,你可以改变一些a[i]的值,改变后,程序还能针对改
变后的a继续回答上面的问题。
对于每一次询问,你都需要输出他的答案,每一个输出占单独的一行。
#include<set> #include<map> #include<queue> #include<cmath> #include<stack> #include<vector> #include<cstdio> #include<cstring> #include<iostream> #include<algorithm> typedef long long ll; using namespace std; int n,m; char ch[2]; int x,y,z; int cnt; int tot; int num; int ls[8000010]; int rs[8000010]; int sum[8000010]; int root[100010]; int s[100010]; int t[100010]; int a[100010]; int updata(int pre,int l,int r,int k) { int rt=++cnt; if(l==r) { sum[rt]=sum[pre]+1; return rt; } ls[rt]=ls[pre]; rs[rt]=rs[pre]; sum[rt]=sum[pre]+1; int mid=(l+r)>>1; if(k<=mid) { ls[rt]=updata(ls[pre],l,mid,k); } else { rs[rt]=updata(rs[pre],mid+1,r,k); } return rt; } int downdata(int pre,int l,int r,int k) { int rt=++cnt; if(l==r) { sum[rt]=sum[pre]-1; return rt; } ls[rt]=ls[pre]; rs[rt]=rs[pre]; sum[rt]=sum[pre]-1; int mid=(l+r)>>1; if(k<=mid) { ls[rt]=downdata(ls[pre],l,mid,k); } else { rs[rt]=downdata(rs[pre],mid+1,r,k); } return rt; } void add(int v,int x) { for(int i=x;i<=n;i+=i&-i) { root[i]=updata(root[i],0,1e9,v); } } void del(int v,int x) { for(int i=x;i<=n;i+=i&-i) { root[i]=downdata(root[i],0,1e9,v); } } int query(int l,int r,int k) { if(l==r) { return l; } int ans=0; int res=0; for(int i=1;i<=num;i++) { res+=sum[ls[s[i]]]; } for(int i=1;i<=tot;i++) { ans+=sum[ls[t[i]]]; } int mid=(l+r)>>1; if(ans-res>=k) { for(int i=1;i<=num;i++) { s[i]=ls[s[i]]; } for(int i=1;i<=tot;i++) { t[i]=ls[t[i]]; } return query(l,mid,k); } else { for(int i=1;i<=num;i++) { s[i]=rs[s[i]]; } for(int i=1;i<=tot;i++) { t[i]=rs[t[i]]; } return query(mid+1,r,k-(ans-res)); } } void ask(int l,int r,int x) { num=0; tot=0; for(int i=l;i;i-=i&-i) { s[++num]=root[i]; } for(int i=r;i;i-=i&-i) { t[++tot]=root[i]; } printf("%d\n",query(0,1e9,x)); } int main() { scanf("%d%d",&n,&m); for(int i=1;i<=n;i++) { scanf("%d",&a[i]); add(a[i],i); } for(int i=1;i<=m;i++) { scanf("%s",ch); if(ch[0]==‘Q‘) { scanf("%d%d%d",&x,&y,&z); ask(x-1,y,z); } else { scanf("%d%d",&x,&y); del(a[x],x); a[x]=y; add(a[x],x); } } }
BZOJ1901 Zju2112 Dynamic Rankings——主席树+树状数组
原文:https://www.cnblogs.com/Khada-Jhin/p/9513584.html