Link:

BZOJ 1901 传送门

Solution:

带修改主席树的模板题

对于静态区间第$k$大直接上主席树就行了

但加上修改后会发现修改时复杂度不满足要求了:

去掉/增加第$i$位上的值时要更新$i...n$间所有的主席树,使得单次修改的复杂度达到$n*log(n)$

可以将原来的主席树看成前缀数组

求某一段时尚可直接差分,但涉及到修改时就要改动$O(n)$级别的节点了

这时想到优化前缀和问题的树状数组

如果将原来的每一棵主席树看作树状数组上的点并利用$lowbit()$修改/求值

这样就能每次改动$log(n)$棵主席树,从而将复杂度降到$log(n)^2$

实现中先离散化,对于每一次修改先去掉删除原值,再添加新值就好啦

注意修改时要先记录所有需要的节点并一起移动,对于$k$大问题无法单独计算

Tip:

1、此时每棵线段树已经不再具有主席树的性质了:每次修改在前者基础上增加一条链

其实现在每棵线段树就是在自己原基础上修改,准确地说就是树状数组套动态开点权值线段树

2、好像此类动态开点线段树的空间复杂度我不太会算……

此题好像$O(n*log(n))$就够用了……

Code:

#include <bits/stdc++.h>

using namespace std;
const int MAXN=;
char s[];
struct Query{int i,j,k;}q[MAXN];
//内存开大,好像这题n*log(n)就够了
struct PrTree{int ls,rs,cnt;}seg[];
int n,m,dat[MAXN],rt[MAXN],L[],R[],dsp[MAXN<<],tot,totl,totr,cnt;
inline int lowbit(int x){return x&(-x);}
//PrTree
void Update(int &cur,int pos,int val,int l,int r)
{
if(!cur) cur=++cnt;
seg[cur].cnt+=val;
if(l==r) return;int mid=(l+r)>>;
if(pos<=mid) Update(seg[cur].ls,pos,val,l,mid);
else Update(seg[cur].rs,pos,val,mid+,r);
} int Query(int k,int l,int r)
{
if(l==r) return l;
int sum=,mid=(l+r)>>;
for(int i=;i<=totl;i++) sum-=seg[seg[L[i]].ls].cnt;
for(int i=;i<=totr;i++) sum+=seg[seg[R[i]].ls].cnt;
if(sum>=k)
{
for(int i=;i<=totl;i++) L[i]=seg[L[i]].ls;
for(int i=;i<=totr;i++) R[i]=seg[R[i]].ls;
return Query(k,l,mid);
}
else
{
for(int i=;i<=totl;i++) L[i]=seg[L[i]].rs;
for(int i=;i<=totr;i++) R[i]=seg[R[i]].rs;
return Query(k-sum,mid+,r);
}
}
//BIT
void upd(int x,int val)
{
int pos=lower_bound(dsp+,dsp+tot+,dat[x])-dsp;
for(int i=x;i<=n;i+=lowbit(i))
Update(rt[i],pos,val,,tot);
} int qry(int l,int r,int k)
{
totl=totr=;
for(int i=l-;i;i-=lowbit(i)) L[++totl]=rt[i];
for(int i=r;i;i-=lowbit(i)) R[++totr]=rt[i];
return dsp[Query(k,,tot)];
} int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
scanf("%d",&dat[i]),dsp[++tot]=dat[i];
for(int i=;i<=m;i++)
{
scanf("%s%d%d",s,&q[i].i,&q[i].j);
if(s[]=='Q') scanf("%d",&q[i].k);
else dsp[++tot]=q[i].j;
}
sort(dsp+,dsp+tot+);
tot=unique(dsp+,dsp+tot+)-dsp-; for(int i=;i<=n;i++) upd(i,);
for(int i=;i<=m;i++)
{
if(q[i].k) printf("%d\n",qry(q[i].i,q[i].j,q[i].k));
else upd(q[i].i,-),dat[q[i].i]=q[i].j,upd(q[i].i,);
}
return ;
}

[BZOJ 1901] Dynamic Rankings的更多相关文章

  1. bzoj 1901 Dynamic Rankings (树状数组套线段树)

    1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec  Memory Limit: 128 MB Description 给定一个含有n个数的序列a[1] ...

  2. [BZOJ 1901] Dynamic Rankings 【树状数组套线段树 || 线段树套线段树】

    题目链接:BZOJ - 1901 题目分析 树状数组套线段树或线段树套线段树都可以解决这道题. 第一层是区间,第二层是权值. 空间复杂度和时间复杂度均为 O(n log^2 n). 线段树比树状数组麻 ...

  3. BZOJ.1901.Dynamic Rankings(线段树套平衡树 Splay)

    题目链接or Here 题意:n个数,有两个操作:1.修改某个数为v:2.询问一段区间第k小的数 如果没有修改,则可以用线段树,每个节点P[a,b]存储大小为b-a+1的数组,代表其中的数 同时,这个 ...

  4. BZOJ.1901.Dynamic Rankings(树状数组套主席树(动态主席树))

    题目链接 BZOJ 洛谷 区间第k小,我们可以想到主席树.然而这是静态的,怎么支持修改? 静态的主席树是利用前缀和+差分来求解的,那么对于每个位置上的每棵树看做一个点,拿树状数组更新. 还是树状数组的 ...

  5. BZOJ.1901.Dynamic Rankings(整体二分)

    题目链接 BZOJ 洛谷 (以下是口胡) 对于多组的询问.修改,我们可以发现: 假设有对p1,p2,p3...的询问,在这之前有对p0的修改(比如+1),且p0<=p1,p2,p3...,那么我 ...

  6. BZOJ 1901 Dynamic Rankings 树董事长

    标题效果:间隔可以改变k少 我的两个天树牌主席... 隔断Count On A Tree 之后我一直认为,随着树的主席的变化是分域林木覆盖率可持久段树. .. 事实上,我是误导... 尼可持久化线段树 ...

  7. BZOJ 1901 Dynamic Rankings (整体二分+树状数组)

    题目大意:略 洛谷传送门 这道题在洛谷上数据比较强 貌似这个题比较常见的写法是树状数组套主席树,动态修改 我写的是整体二分 一开始的序列全都视为插入 对于修改操作,把它拆分成插入和删除两个操作 像$C ...

  8. bzoj 1901: Zju2112 Dynamic Rankings(树套树)

    1901: Zju2112 Dynamic Rankings 经典的带改动求区间第k小值问题 树套树模板,我是用的线段树套splay实现的,并且用的数组模拟的,所以可能空间略大,bzoj过了,zoj过 ...

  9. BZOJ 1901 Zju2112 Dynamic Rankings

    树阵主席设置树.维护间隔动态K大. .. ZOJ到空间太小,太大,仅仅能到BZOJ上交 1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec  Memor ...

随机推荐

  1. 将资源文件夹中的文件通过流的方式写入到应用的File文件夹中

    //1.在Files文件夹中创建同名的数据库文件 File files = getFilesDir(); File file = new File(files, DBName); if(file.ex ...

  2. linux安装lamp

    github https://github.com/zblogcn/zblogphp Installation If your server system: CentOS yum -y install ...

  3. github新建托管项目及上传项目

    一.新建托管项目 1.注册: 2.点击new repositories新建一个新项目: 3.输入项目名称及项目描述,Create repository: 4.点击右边栏的剪切板图标,记录下你的项目地址 ...

  4. ImportError: libQtTest.so.4: cannot open shared

    错误: import cv2 File , in <module> from .cv2 import * ImportError: libQtTest.so.: cannot open s ...

  5. 【DeepLearning学习笔记】Coursera课程《Neural Networks and Deep Learning》——Week1 Introduction to deep learning课堂笔记

    Coursera课程<Neural Networks and Deep Learning> deeplearning.ai Week1 Introduction to deep learn ...

  6. python基础===新式类与经典类

    首先: Python 2.x中默认都是经典类,只有显式继承了object才是新式类 Python 3.x中默认都是新式类,不必显式的继承object 这两种类的区别: 新式类重定义的方法更多,当然这不 ...

  7. ADO POST时出现“无法为更新定位行,一些值可能已在最后一次读取后已更改”问题的解决方法

    原因有这样几种: 1.在数据库设计时,为某些字段设置了默认值,在修改进行提交以后,数据库会自动修改对应字段的所有行的默认值,从而导致了数据库与数据集中数据的不一致,使ADOQuery无法对数据集进行定 ...

  8. linux运维记

    nmap 127.0.0.1 命令查看当前服务器对外有多少端口,用于检查漏洞 vim ctrl+z ,jobs,fg 切换控制应用程序 vim 执行命令 #!sh aa.sh执行命令 运维系统监控开源 ...

  9. 算法题之Median of Two Sorted Arrays

    这道题是LeetCode上的题目,难度级别为5,刚开始做没有找到好的思路,以为是自己智商比较低,后来发现确实也比较低... 题目: There are two sorted arrays nums1  ...

  10. tornado 模版

    tornado 模版语法 取消转义 : 取消项目转义 :autoescape = None 取消模版转义:{% autoescape None %} 取消行转义   :{% raw bd %} 强制转 ...