bzoj 1901: Zju2112 Dynamic Rankings(树套树)
1901: Zju2112 Dynamic Rankings
经典的带改动求区间第k小值问题
树套树模板,我是用的线段树套splay实现的,并且用的数组模拟的,所以可能空间略大,bzoj过了,zoj过不了。
思路非常easy,用线段树维护区间,用splay维护区间内的权值,然后询问的时候,二分答案key,然后在区间内找小于key的数有多少个。
贴上模板:
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define lson l , m , rt<<1
#define rson m + 1 , r , rt<<1|1
#define ls son[0][rt]
#define rs son[1][rt]
using namespace std ; const int maxn = 100001 ;
const int N = maxn*18 ;
const int INF = 1111111111 ;
int a[maxn] , n , pool[N] , top ;
struct TTT {
int son[2][N] , size[N] , fa[N] , val[N] ;
int root[N] , tot ;
void push_up ( int rt ) {
size[rt] = size[ls] + size[rs] + 1 ;
}
int new_node ( int _val ) {
int rt ;
if ( top ) rt = pool[top--] ;
else rt = ++tot ;
size[rt] = 1 ;
ls = rs = fa[rt] = 0 ;
val[rt] = _val ;
return rt ;
}
void build_root ( int rt ) {
rs = new_node ( INF ) ;
fa[rs] = rt ;
}
void build ( int l , int r , int rt ) {
root[rt] = new_node(-INF) ;
build_root ( root[rt] ) ;
if ( l == r ) return ;
int m = l + r >> 1 ;
build ( lson ) ;
build ( rson ) ;
}
void rot ( int rt ) {
int y = fa[rt] , z = fa[y] , c = rt == son[0][y] ;
son[!c][y] = son[c][rt] , fa[son[c][rt]] = y ;
son[c][rt] = y , fa[y] = rt ;
fa[rt] = z ;
son[y==son[1][z]][z] = rt ;
push_up ( y ) ;
}
void splay ( int rt , int to ) {
while ( fa[rt] != to ) {
int y = fa[rt] , z = fa[y] ;
if ( z != to ) rot ( (rt==son[0][y])^(y==son[0][z]) ? rt : y ) ;
rot ( rt ) ;
}
push_up ( rt ) ;
}
int find_smaller ( int rt , int v ) {
int ret ;
if ( val[rt] >= v ) {
if ( !ls ) return rt ;
ret = find_smaller ( ls , v ) ;
} else {
if ( !rs ) return rt ;
ret = find_smaller ( rs , v ) ;
if ( val[ret] >= v ) ret = rt ;
}
return ret ;
}
int find_left_most ( int rt ) {
while ( ls ) rt = ls ;
return rt ;
}
int count_smaller ( int rt , int v ) {
if ( !rt ) return 0 ;
if ( val[rt] < v ) return size[ls] + 1 + count_smaller ( rs , v ) ;
else return count_smaller ( ls , v ) ;
}
void add ( int& rt , int v ) {
rt = find_smaller ( rt , v ) ;
splay ( rt , 0 ) ;
int temp = find_left_most ( rs ) ;
splay ( temp , rt ) ;
son[0][temp] = new_node ( v ) ;
fa[son[0][temp]] = temp ;
push_up ( temp ) ; push_up ( rt ) ;
}
void print ( int rt ) {
if ( !rt ) return ;
printf ( "rt = %d , sz = %d\n" , rt , size[rt] ) ;
printf ( "ls = %d , rs = %d , fa = %d , val = %d\n" , ls , rs , fa[rt] , val[rt] ) ;
print ( ls ) ;
print ( rs ) ;
}
void del ( int& rt , int v ) {
rt = find_smaller ( rt , v ) ;
splay ( rt , 0 ) ;
int temp = find_left_most ( rs ) ;
if ( val[temp] != v ) return ;
splay ( temp , rt ) ;
temp = find_left_most ( son[1][temp] ) ;
splay ( temp , rt ) ;
pool[++top] = son[0][temp] ;
son[0][temp] = 0 ;
push_up ( temp ) ; push_up ( rt ) ;
}
void update ( int pos , int v , int op , int l , int r , int rt ) {
if ( op == 0 ) {
add ( root[rt] , v ) ;
}
else del ( root[rt] , v ) ;
if ( l == r ) return ;
int m = l + r >> 1 ;
if ( pos <= m ) update ( pos , v , op , lson ) ;
else update ( pos , v , op , rson ) ;
}
int count ( int a , int b , int v , int l , int r , int rt ) {
if ( a <= l && r <= b ) {
int k = count_smaller ( root[rt] , v ) - 1 ;
return k ;
}
int m = l + r >> 1 , ret = 0 ;
if ( a <= m ) ret = count ( a , b , v , lson ) ;
if ( m < b ) ret += count ( a , b , v , rson ) ;
return ret ;
}
void init () {
tot = 0 ;
top = 0 ;
build ( 1 , n , 1 ) ;
for ( int i = 1 ; i <= n ; i ++ ) {
// printf ( "a[%d] = %d\n" , i , a[i] ) ;
update ( i , a[i] , 0 , 1 , n , 1 ) ;
}
}
int ask ( int a , int b , int k ) {
int l = 0 , r = 1000000000 ;
while ( l <= r ) {
int m = l + r >> 1 ;
int cnt = count ( a , b , m , 1 , n , 1 ) ;
// printf ( "m = %d , cnt = %d\n" , m , cnt ) ;
if ( cnt <= k - 1 ) l = m + 1 ;
else r = m - 1 ;
// printf ( "l = %d , r = %d\n" , l , r ) ;
}
return r ;
}
void modify ( int pos , int num ) {
update ( pos , a[pos] , 1 , 1 , n , 1 ) ;
update ( pos , num , 0 , 1 , n , 1 ) ;
a[pos] = num ;
}
} ttt ;
int main () {
int T , m ;
// freopen ( "a.in" , "r" , stdin ) ;
// scanf ( "%d" , &T ) ;
while ( scanf ( "%d%d" , &n , &m ) != EOF ) {
for ( int i = 1 ; i <= n ; i ++ )
scanf ( "%d" , &a[i] ) ;
ttt.init () ;
while ( m -- ) {
char op[11] ;
int x , y , z ;
scanf ( "%s" , op ) ;
if ( op[0] == 'Q' ) {
scanf ( "%d%d%d" , &x , &y , &z ) ;
printf ( "%d\n" , ttt.ask ( x , y , z ) ) ;
} else {
scanf ( "%d%d" , &x , &y ) ;
ttt.modify ( x , y ) ;
}
}
}
return 0 ;
}
bzoj 1901: Zju2112 Dynamic Rankings(树套树)的更多相关文章
- BZOJ 1901: Zju2112 Dynamic Rankings( BIT 套 BST )
BIT 套 splay 其实也是不难...每个 BIT 的结点保存一颗 splay , 询问就二分答案然后判断rank... ------------------------------------- ...
- Bzoj 1901: Zju2112 Dynamic Rankings 树套树,线段树,平衡树,Treap
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 6471 Solved: 2697[Su ...
- BZOJ 1901: Zju2112 Dynamic Rankings( 树状数组套主席树 )
裸的带修改主席树.. 之前用BIT套Splay( http://www.cnblogs.com/JSZX11556/p/4625552.html )A过..但是还是线段树好写...而且快(常数比平衡树 ...
- BZOJ 1901: Zju2112 Dynamic Rankings[带修改的主席树]【学习笔记】
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 7143 Solved: 2968[Su ...
- Bzoj 1901: Zju2112 Dynamic Rankings 主席树,可持久,树状数组,离散化
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 6321 Solved: 2628[Su ...
- bzoj 1901: Zju2112 Dynamic Rankings -- 主席树,树状数组,哈希
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MB Description 给定一个含有n个数的序列a[1] ...
- BZOJ 1901 Zju2112 Dynamic Rankings
树阵主席设置树.维护间隔动态K大. .. ZOJ到空间太小,太大,仅仅能到BZOJ上交 1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memor ...
- BZOJ 1901: Zju2112 Dynamic Rankings 区间k大 带修改 在线 线段树套平衡树
之前写线段树套splay数组版..写了6.2k..然后弃疗了.现在发现还是很水的..嘎嘎.. zju过不了,超时. upd:才发现zju是多组数据..TLE一版才发现.然后改了,MLE...手写内存池 ...
- BZOJ 1901 Zju2112 Dynamic Rankings 树状数组套线段树
题意概述:带修改求区间第k大. 分析: 我们知道不带修改的时候直接上主席树就可以了对吧?两个版本号里面的节点一起走在线段树上二分,复杂度是O((N+M)logN). 然而这里可以修改,主席树显然是凉了 ...
随机推荐
- 【原创】MapGIS K9 三维二次开发入门
开发语言:C# 平台版本:MapGIS K9 SP3 MapGIS K9三维平台也提供了接口和组件以实现二次开发.用户可以根据提供的接口和组件进行二次开发,也可以借助MapGISK9数据中心框架,可以 ...
- REF CURSOR和CURSOR
REF CURSOR DECLARE TYPE TY_EMP_CUR IS REF CURSOR; V_Emp_Cur TY_EMP_CUR; V_Id EMP.ID%TYPE; BEGIN OPEN ...
- dede修改移动文档的js
dede后台弹框修改: 想做个类似文章列表的移动功能,弹框,然后修改成功到表 先在list.js里复制一份moveArc的方法,到archives_do.php里复制一份moveArchives的方法 ...
- 在verilog中关于inout口的设计方法
在学习IIC的时候我们知道这么设计inout inout scl : reg scl_reg , scl_en ; scl = scl_en ? scl_reg : 1'dz ; 当 ...
- Java InputStream读取网络响应Response数据的方法
Java InputStream读取数据问题 原理讲解 1. 关于InputStream.read() 在从数据流里读取数据时,为图简单,经常用InputStream.read()方法.这个方 ...
- Windows Phone 8初学者开发—第3部分:编写第一个Windows Phone 8应用程序
原文 Windows Phone 8初学者开发—第3部分:编写第一个Windows Phone 8应用程序 原文地址: http://channel9.msdn.com/Series/Windows- ...
- uvalive 2322 Wooden Sticks(贪心)
题目连接:2322 Wooden Sticks 题目大意:给出要求切的n个小木棍 , 每个小木棍有长度和重量,因为当要切的长度和重量分别大于前面一个的长度和重量的时候可以不用调整大木棍直接切割, 否则 ...
- MFC基础类源码CPP实现文件
WinMain.CPP---->AfxWinMain() //近似可认为是WinMain()函数的入口 VIEWCORE.CPP---->CView DOCCORE.CPP----> ...
- 介绍一款开源的类Excel电子表格软件
Excel一直以霸主的地位,占领了Windows桌面表格软件市场No 1,与此同一时候,Office套装产品差点儿成为了IT行业的标配办公技能.有无相似Excel的桌面程序,绿色版,实现主要的数 ...
- android 4.2 root
前一段因工作需要,对android4.2 进行root.但是在下载了 点击打开链接,下载了Superuser.apk,把对应的apk拷贝到system/app,su拷贝到/system/bin 与/s ...