主席树带修第k大 
https://www.cnblogs.com/Empress/p/4659824.html 讲的非常好的博客

首先按静态第k大建立起一组权值线段树(主席树)
然后现在要将第i个值从a改为b,
由主席树前缀和的性质可知修改第i个值会对T[i]...T[n]棵权值线段树造成相同的影响
如果暴力进行维护必定超时
那么如何优化这种修改? 我们知道动态维护前缀和可以用bit在log时间内完成
那么对于动态维护具有前缀和性质的主席树,我们也可以套上bit来完成
update:将第i个值从a改成b
根据bit的原理,第i个元素的修改会影响第i+
lowbit(i)个元素
那么现在第i棵树修改了,第i+lowbit(i)棵树也应该影响
即第i棵树的位置a -1,同样的第i+lowbit(i)棵树的位置a -1
同理这批树的位置b +1即可
query:询问区间[l,r]第k大元素
即求T[r]-
T[l]的前缀和即可
查询时要把bit对应的修改套上

一些实现上的细节

  按照静态主席树建立 T[1-n]
  用相同结构的树状数组 S[1-n]来维护 T[1-n]
  S[i]表示一棵权值线段树的根节点

  update:修改位置i,必然要修改S[i],那么S[i+lowbit i]也要修改
  query [l,r]:转化成前缀和问题,静态的初始主席树 T[r]-T[l-1], 再加上树状数组中修改的量 S[r]-S[l-1],

由于需要到子树里更新或查询,开一个数组use[i]表示第i棵树状数组当前在哪个子树更新或查询

我写的不知道为什么在zoj会段错误。。

在洛谷上就能过。。

#include<bits/stdc++.h>
using namespace std;
#define maxn 600005
inline int low(int x){return x&(-x);}
struct Q{char op[];int l,r,k;}op[maxn];
int n,a[maxn],m,tot,b[maxn],q; struct Node{int lc,rc,sum;}T[maxn*];
int rt[maxn],S[maxn],use[maxn],size;
int build(int l,int r){
int now=++size;
if(l==r)return now;
int mid=l+r>>;
T[now].lc=build(l,mid);
T[now].rc=build(mid+,r);
return now;
}
int update(int last,int pos,int v,int l,int r){
int now=++size;
T[now]=T[last];T[now].sum+=v;
if(l==r)return now;
int mid=l+r>>;
if(pos<=mid)T[now].lc=update(T[last].lc,pos,v,l,mid);
else T[now].rc=update(T[last].rc,pos,v,mid+,r);
return now;
}
int Sum(int x){//查询前x棵树的修改值
int res=;
while(x){
res+=T[T[use[x]].lc].sum;
x-=low(x);
}
return res;
}
void Update(int x,int pos,int v){
while(x<=n){
S[x]=update(S[x],pos,v,,m);
x+=low(x);
}
}
int query(int L,int R,int st,int ed,int l,int r,int k){
if(l==r)return l;
int mid=l+r>>;
int sum=Sum(R)-Sum(L)+T[T[ed].lc].sum-T[T[st].lc].sum;
if(k<=sum){
for(int i=L;i;i-=low(i))
use[i]=T[use[i]].lc;
for(int i=R;i;i-=low(i))
use[i]=T[use[i]].lc;
return query(L, R, T[st].lc, T[ed].lc,l,mid,k);
}
else {
for(int i=L;i;i-=low(i))
use[i]=T[use[i]].rc;
for(int i=R;i;i-=low(i))
use[i]=T[use[i]].rc;
return query(L, R, T[st].rc, T[ed].rc,mid+,r, k-sum);
}
}
void init(){
m=tot=size=;memset(rt,,sizeof rt);
memset(S,,sizeof S);
}
int main(){ {
init();
scanf("%d%d",&n,&q);
for(int i=;i<=n;i++)scanf("%d",&a[i]),b[++tot]=a[i];
for(int i=;i<=q;i++){
scanf("%s",op[i].op);
if(op[i].op[]=='Q')scanf("%d%d%d",&op[i].l,&op[i].r,&op[i].k);
else {
scanf("%d%d",&op[i].l,&op[i].r);
b[++tot]=op[i].r;
}
}
sort(b+,b++tot);//离散化
m=unique(b+,b++tot)-(b+); rt[]=build(,m);
for(int i=;i<=n;i++){//建立静态主席树
int pos=lower_bound(b+,b++m,a[i])-b;
rt[i]=update(rt[i-],pos,,,m);
S[i]=rt[];
}
for(int i=;i<=q;i++){
int l=op[i].l,r=op[i].r,k=op[i].k;
if(op[i].op[]=='Q'){//查询,准备好要用到的线段树
for(int j=l-;j;j-=low(j))use[j]=S[j];
for(int j=r;j;j-=low(j))use[j]=S[j];
cout<<b[query(l-,r,rt[l-],rt[r],,m,k)]<<'\n';
}
else {
int pos1=lower_bound(b+,b++m,a[l])-b;
int pos2=lower_bound(b+,b++m,r)-b;
Update(l,pos1,-);Update(l,pos2,);
a[l]=r;
}
}
}
}

下面是一份模板。。

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
#define lson l, m
#define rson m+1, r
const int N=;
int a[N], Hash[N];
int T[N], L[N<<], R[N<<], sum[N<<];
int S[N];
int n, m, tot;
struct node
{
int l, r, k;
bool Q;
}op[]; int build(int l, int r)
{
int rt=(++tot);
sum[rt]=;
if(l!=r)
{
int m=(l+r)>>;
L[rt]=build(lson);
R[rt]=build(rson);
}
return rt;
} int update(int pre, int l, int r, int x, int val)
{
int rt=(++tot);
L[rt]=L[pre], R[rt]=R[pre], sum[rt]=sum[pre]+val;
if(l<r)
{
int m=(l+r)>>;
if(x<=m)
L[rt]=update(L[pre], lson, x, val);
else
R[rt]=update(R[pre], rson, x, val);
}
return rt;
} int lowbit(int x)
{
return x&(-x);
} int use[N];
void add(int x, int pos, int val)
{
while(x<=n)
{
S[x]=update(S[x], , m, pos, val);
x+=lowbit(x);
}
} int Sum(int x)
{
int ret=;
while(x>)
{
ret+=sum[L[use[x]]];
x-=lowbit(x);
}
return ret;
} int query(int u, int v, int lr, int rr, int l, int r, int k)
{
if(l>=r)
return l;
int m=(l+r)>>;
int tmp=Sum(v)-Sum(u)+sum[L[rr]]-sum[L[lr]];
if(tmp>=k)
{
for(int i=u;i;i-=lowbit(i))
use[i]=L[use[i]];
for(int i=v;i;i-=lowbit(i))
use[i]=L[use[i]];
return query(u, v, L[lr], L[rr], lson, k);
}
else
{
for(int i=u;i;i-=lowbit(i))
use[i]=R[use[i]];
for(int i=v;i;i-=lowbit(i))
use[i]=R[use[i]];
return query(u, v, R[lr], R[rr], rson, k-tmp);
}
} void modify(int x, int p, int d)
{
while(x<=n)
{
S[x]=update(S[x], , m, p, d);
x+=lowbit(x);
}
} int main()
{
int t;
scanf("%d", &t);
while(t--)
{
int q;
scanf("%d%d", &n, &q);
tot=;
m=;
for(int i=;i<=n;i++)
{
scanf("%d", &a[i]);
Hash[++m]=a[i];
}
for(int i=;i<q;i++)
{
char s[];
scanf("%s", s);
if(s[]=='Q')
{
scanf("%d%d%d", &op[i].l, &op[i].r, &op[i].k);
op[i].Q=;
}
else
{
scanf("%d%d", &op[i].l, &op[i].r);
op[i].Q=;
Hash[++m]=op[i].r;
}
}
sort(Hash+, Hash++m);
int mm=unique(Hash+, Hash++m)-Hash-;
m=mm;
T[]=build(, m);
for(int i=;i<=n;i++)
T[i]=update(T[i-], , m, lower_bound(Hash+, Hash++m, a[i])-Hash, );
for(int i=;i<=n;i++)
S[i]=T[];
for(int i=;i<q;i++)
{
if(op[i].Q)
{
for(int j=op[i].l-;j;j-=lowbit(j))
use[j]=S[j];
for(int j=op[i].r;j;j-=lowbit(j))
use[j]=S[j];
printf("%d\n", Hash[query(op[i].l-, op[i].r, T[op[i].l-], T[op[i].r], , m, op[i].k)]);
}
else
{
modify(op[i].l, lower_bound(Hash+, Hash++m, a[op[i].l])-Hash, -);
modify(op[i].l, lower_bound(Hash+, Hash++m, op[i].r)-Hash, );
a[op[i].l]=op[i].r;
}
}
}
return ;
}

主席树套树状数组——带修区间第k大zoj2112的更多相关文章

  1. 【ZOJ2112】【整体二分+树状数组】带修改区间第k大

    The Company Dynamic Rankings has developed a new kind of computer that is no longer satisfied with t ...

  2. luogu P2617 Dynamic Rankings && bzoj 1901 (带修改区间第k大)

    链接:https://www.luogu.org/problemnew/show/P2617 思路: 如果直接在主席树上修改的话,每次修改都会对后面所有的树造成影响,一次修改的复杂度就会变成 : n* ...

  3. Dynamic Rankings——带修改区间第k大

    三种做法:1.整体二分: 二分mid 考虑小于mid的修改的影响 但是大于mid的修改可能会干掉小于mid的一些值 所以额外把一个修改变成一个值的删除和一个值的添加 这样就相互独立了! 整体二分,树状 ...

  4. 树状数组+二分答案查询第k大的数 (团体程序设计天梯赛 L3-002. 堆栈)

    前提是数的范围较小 1 数据范围:O(n) 2 查第k大的数i:log(n)(树状数组查询小于等于i的数目)*log(n)(二分找到i) 3 添加:log(n) (树状数组) 4 删除:log(n) ...

  5. 静态区间第k大(归并树)

    POJ 2104为例 思想: 利用归并排序的思想: 建树过程和归并排序类似,每个数列都是子树序列的合并与排序. 查询过程,如果所查询区间完全包含在当前区间中,则直接返回当前区间内小于所求数的元素个数, ...

  6. [bzoj3196][Tyvj1730]二逼平衡树_树套树_位置线段树套非旋转Treap/树状数组套主席树/权值线段树套位置线段树

    二逼平衡树 bzoj-3196 Tyvj-1730 题目大意:请写出一个维护序列的数据结构支持:查询给定权值排名:查询区间k小值:单点修改:查询区间内定值前驱:查询区间内定值后继. 注释:$1\le ...

  7. [luogu2617][bzoj1901][Zju2112]Dynamic Rankings【树套树+树状数组+主席树】

    题目网址 [传送门] 题目大意 请你设计一个数据结构,支持单点修改,区间查询排名k. 感想(以下省略脏话inf个字) 真的强力吹爆洛谷数据,一般的树套树还给我T了一般的点,加强的待修主席树还给我卡了几 ...

  8. ZOJ - 2112 主席树套树状数组

    题意:动态第k大,可单点更新,操作+原数组范围6e4 年轻人的第一道纯手工树套树 静态第k大可以很轻易的用权值主席树作差而得 而动态第k大由于修改第i个数会影响[i...n]棵树,因此我们不能在原主席 ...

  9. 「洛谷1903」「BZOJ2120」「国家集训队」数颜色【带修莫队,树套树】

    题目链接 [BZOJ传送门] [洛谷传送门] 题目大意 单点修改,区间查询有多少种数字. 解法1--树套树 可以直接暴力树套树,我比较懒,不想写. 稍微口胡一下,可以直接来一个树状数组套主席树,也就是 ...

随机推荐

  1. vue打包静态资源路径不正确的解决办法【转】

    vue项目完成打包上线的时候很多人都会碰到静态资源找不到的问题,常见的有两个 1.js,css路径不对 解决办法:打开config/index.js,将其中的assetsPublicPath值改为’. ...

  2. [转载] .NET 中可以有类似 JVM 的幻像引用吗?

    近日发现一篇不错的文章,文中列举了一些 GC 场景,探讨了 在 .NET 中是需要实现像 JVM 的中的幻像引用.有人质疑其不切实际,也有像 Ayende 大神一言不合就自己做了个 demo. Do ...

  3. P1238 走迷宫

    原题链接 https://www.luogu.org/problemnew/show/P1238 为了巩固一下刚学习的广搜,练一下迷宫类型的题 不过这道题我用的深搜..... 看问题,我们就知道这道题 ...

  4. 在中断服务函数中使用FreeRTOS系统延时函数vTaskDelay导致看门狗复位的情况

    @2019-04-09 [问题] 控制程序工作一段时间异常重启 [分析] 经定位分析重启原因为看门狗复位导致 [解决] 经排查发现在中断服务函数中使用了FreeRTOS的系统时延函数vTaskDela ...

  5. MySQ数据备份

    MySQL备份概述 问题:备份和冗余有什么区别? 备份:能够防止由于机械故障以及人为操作带来的数据丢失,例如将数据库文件保存在了其它地方. 冗余:数据有多份冗余,但不等于备份,只能防止机械故障带来的数 ...

  6. 小白月赛13 小A与小B (双向BFS)

    链接:https://ac.nowcoder.com/acm/contest/549/G来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言52428 ...

  7. H3C WAP712C 路由器设置

    0.做完任何设置之后都要执行保存操作,否则断电后设置会丢失! 1.默认登录参数:IP:192.168.0.50ID:adminPD:h3capadmin 2.修改默认IP地址:设备 --> 接口 ...

  8. DB(1):SQLAPI catch [Bind variable/parameter 'pay_acc_id' not found] !!!

    SQLAPI catch [Bind variable/parameter 'pay_acc_id' not found] !!! 出现这种报错,先检查命令类后面的参数是否混淆(SACommand s ...

  9. Maven 学习总结 (五) 之 持续集成、构建web应用

    持续集成的作用.过程和优势 简单说,持续集成就是快速且高频率地自动构建项目的所有源码,并为项目成员提供丰富的反馈信息. 快速:集成的速度要尽可能地快,开发人员不希望自己的代码提交半天之后才得到反馈. ...

  10. python2和python3的区别

    python2和python3的区别 参考链接:http://www.runoob.com/python/python-2x-3x.html 1.源码上的区别 python2 python3 源码不规 ...