[BZOJ 1901] Dynamic Rankings
Link:
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的更多相关文章
- bzoj 1901 Dynamic Rankings (树状数组套线段树)
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MB Description 给定一个含有n个数的序列a[1] ...
- [BZOJ 1901] Dynamic Rankings 【树状数组套线段树 || 线段树套线段树】
题目链接:BZOJ - 1901 题目分析 树状数组套线段树或线段树套线段树都可以解决这道题. 第一层是区间,第二层是权值. 空间复杂度和时间复杂度均为 O(n log^2 n). 线段树比树状数组麻 ...
- BZOJ.1901.Dynamic Rankings(线段树套平衡树 Splay)
题目链接or Here 题意:n个数,有两个操作:1.修改某个数为v:2.询问一段区间第k小的数 如果没有修改,则可以用线段树,每个节点P[a,b]存储大小为b-a+1的数组,代表其中的数 同时,这个 ...
- BZOJ.1901.Dynamic Rankings(树状数组套主席树(动态主席树))
题目链接 BZOJ 洛谷 区间第k小,我们可以想到主席树.然而这是静态的,怎么支持修改? 静态的主席树是利用前缀和+差分来求解的,那么对于每个位置上的每棵树看做一个点,拿树状数组更新. 还是树状数组的 ...
- BZOJ.1901.Dynamic Rankings(整体二分)
题目链接 BZOJ 洛谷 (以下是口胡) 对于多组的询问.修改,我们可以发现: 假设有对p1,p2,p3...的询问,在这之前有对p0的修改(比如+1),且p0<=p1,p2,p3...,那么我 ...
- BZOJ 1901 Dynamic Rankings 树董事长
标题效果:间隔可以改变k少 我的两个天树牌主席... 隔断Count On A Tree 之后我一直认为,随着树的主席的变化是分域林木覆盖率可持久段树. .. 事实上,我是误导... 尼可持久化线段树 ...
- BZOJ 1901 Dynamic Rankings (整体二分+树状数组)
题目大意:略 洛谷传送门 这道题在洛谷上数据比较强 貌似这个题比较常见的写法是树状数组套主席树,动态修改 我写的是整体二分 一开始的序列全都视为插入 对于修改操作,把它拆分成插入和删除两个操作 像$C ...
- bzoj 1901: Zju2112 Dynamic Rankings(树套树)
1901: Zju2112 Dynamic Rankings 经典的带改动求区间第k小值问题 树套树模板,我是用的线段树套splay实现的,并且用的数组模拟的,所以可能空间略大,bzoj过了,zoj过 ...
- BZOJ 1901 Zju2112 Dynamic Rankings
树阵主席设置树.维护间隔动态K大. .. ZOJ到空间太小,太大,仅仅能到BZOJ上交 1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memor ...
随机推荐
- Java并发—— 关键字volatile解析
简述 关键字volatile可以说是Java虚拟机提供的最轻量级的同步机制,当一个变量定义为volatile,它具有内存可见性以及禁止指令重排序两大特性,为了更好地了解volatile关键字,我们可以 ...
- GBK UTF-16 UTF-8 编码表
GBK UTF-16 UTF-8 ================== D2BB 4E00 E4 B8 80 一 B6A1 4E01 E4 B8 81 丁 C6DF 4E03 E4 ...
- node启动服务
npm install http-server -g http-server ipconfig查看当前ip 手机可访问第一个网址.
- android隐藏EditText光标
在android中如果有EditText,那么在载入时,光标会默认显示在第一个EditText框中,如果不想显示光标,且也不想把该光标移动到下一个EditText框,最简单的方法是在该 EditTex ...
- iOS APP程序启动原理
UIApplication 程序启动原理 一个应用程序运行就必须要有一个进程,一个进程至少要有一个线程,我们把这个线程叫做主线程,主线程开启之后会开启一个主运行循环,如果不开启一个运行循环,程序开启了 ...
- 设计模式之笔记--工厂方法模式(Factory Method)
工厂方法模式(Factory Method) 定义 工厂方法模式(Factory Method),定义一个用于创建对象的接口,让子类决定实例化哪一个类.工厂方法使一个类的实例化延迟到其子类. 类图 描 ...
- 排名函数row_number() over(order by)用法
1. 定义 简单的说row_number()从1开始,为每一条分组记录返回一个数字,这里的ROW_NUMBER() OVER (ORDER BY [列名]DESC) 是先把[列名]降序排列,再为降序以 ...
- Linux操作系统中内存buffer和cache的区别--从free命令说起(转)
原文链接:http://os.51cto.com/art/200709/56603.htm 我们一开始,先从Free命令说起. Free free 命令相对于top 提供了更简洁的查看系统内存使用情况 ...
- 深度学习方法:受限玻尔兹曼机RBM(一)基本概念
欢迎转载,转载请注明:本文出自Bin的专栏blog.csdn.net/xbinworld. 技术交流QQ群:433250724,欢迎对算法.技术.应用感兴趣的同学加入. 最近在复习经典机器学习算法的同 ...
- 【hdoj_2189】来生一起走(母函数)
题目:http://acm.hdu.edu.cn/showproblem.php?pid=2189 本题的数学模型如下: 分解的问题,常用母函数求解,这里要求每个"硬币"的价值必须 ...