BZOJ3196——二逼平衡树
1、题目大意:给你一个序列,有5种操作,都有什么呢。。。
1> 区间第k小 这个直接用二分+树套树做
2> 区间小于k的有多少 这个直接用树套树做
3> 单点修改 这个直接用树套树做
4> 区间内k的前驱 这个就是1和2操作的合并,就是查询k的排名,然后就是知道他的前驱的排名,然后第k小
5> 区间内k的后继 这个和4同理
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
using namespace std;
struct Node{
Node *ch[2];
int num, cnt, v, r;
bool operator < (const Node& rhs) const{
return r < rhs.r;
}
inline int cmp(int x){
if(x == v) return -1;
if(x < v) return 0;
return 1;
}
inline void maintain(){
cnt = num;
if(ch[0]) cnt += ch[0] -> cnt;
if(ch[1]) cnt += ch[1] -> cnt;
return;
}
} ft[5000000], *root[400000];
int tot;
int a[100000];
inline void treap_rotate(Node* &o, int d){
Node *k = o -> ch[d ^ 1];
o -> ch[d ^ 1] = k -> ch[d];
k -> ch[d] = o;
o -> maintain();
k -> maintain();
o = k;
return;
}
inline void treap_insert(Node* &o, int value){
if(!o){
o = &ft[tot ++];
o -> ch[0] = o -> ch[1] = NULL;
o -> num = 1;
o -> v = value;
o -> r = rand();
}
else{
int d = o -> cmp(value);
if(d == -1){
o -> num ++;
}
else{
treap_insert(o -> ch[d], value);
if(o -> ch[d] > o) treap_rotate(o, d ^ 1);
}
}
o -> maintain();
}
inline void treap_remove(Node* &o, int value){
if(!o) return;
int d = o -> cmp(value);
if(d == -1){
if(o -> num > 1) o -> num --;
else if(!o -> ch[0]) o = o -> ch[1];
else if(!o -> ch[1]) o = o -> ch[0];
else{
int d2;
if(o -> ch[0] > o -> ch[1]) d2 = 1;
else d2 = 0;
treap_rotate(o, d2);
treap_remove(o -> ch[d2], value);
}
}
else{
treap_remove(o -> ch[d], value);
}
if(o) o -> maintain();
}
inline int treap_lessk(Node* &o, int k){
if(!o) return 0;
int d = o -> cmp(k);
if(d == -1){
int ret = o -> num;
if(o -> ch[0]) ret += o -> ch[0] -> cnt;
return ret;
}
else if(d == 0){
return treap_lessk(o -> ch[d], k);
}
else{
int ss = o -> num;
if(o -> ch[0]) ss += o -> ch[0] -> cnt;
return treap_lessk(o -> ch[d], k) + ss;
}
}
inline int treap_find(Node* &o, int k){
if(!o) return 0;
int d = o -> cmp(k);
if(d == -1) return o -> num;
else return treap_find(o -> ch[d], k);
}
inline int segment_tree_find(int l, int r, int o, int x, int y, int k){
if(x <= l && r <= y){
return treap_find(root[o], k);
}
int mid = (l + r) / 2;
int ret = 0;
if(x <= mid) ret += segment_tree_find(l, mid, 2 * o, x, y, k);
if(y > mid) ret += segment_tree_find(mid + 1, r, 2 * o + 1, x, y, k);
return ret;
}
inline void segment_tree_add(int l, int r, int o, int x, int value){
treap_remove(root[o], a[x]);
treap_insert(root[o], value);
if(l == r){
a[x] = value;
return;
}
int mid = (l + r) / 2;
if(x <= mid) segment_tree_add(l, mid, 2 * o, x, value);
else segment_tree_add(mid + 1, r, 2 * o + 1, x, value);
}
inline int segment_tree_query_lessk(int l, int r, int o, int x, int y, int k){
if(x <= l && r <= y){
return treap_lessk(root[o], k);
}
int mid = (l + r) / 2;
int ret = 0;
if(x <= mid) ret += segment_tree_query_lessk(l, mid, 2 * o, x, y, k);
if(y > mid) ret += segment_tree_query_lessk(mid + 1, r, 2 * o + 1, x, y, k);
return ret;
}
inline int segment_tree_query_NO_k(int l, int r, int o, int x, int y, int k, int n){
int L = 0, R = 100000000;
while(L < R){
int mid = (L + R) / 2;
if(segment_tree_query_lessk(1, n, 1, x, y, mid) < k) L = mid + 1;
else R = mid;
}
return L;
}
int main(){
int n, m;
scanf("%d%d", &n, &m);
for(int i = 1; i <= n; i ++){
int x;
scanf("%d", &x);
segment_tree_add(1, n, 1, i, x);
}
for(int i = 1; i <= m; i ++){
int op, x, y, z;
scanf("%d", &op);
if(op == 1){
scanf("%d%d%d", &x, &y, &z);
int ans = segment_tree_query_lessk(1, n, 1, x, y, z);
ans -= segment_tree_find(1, n, 1, x, y, z);
ans ++;
printf("%d\n", ans);
}
else if(op == 2){
scanf("%d%d%d", &x, &y, &z);
int ans = segment_tree_query_NO_k(1, n, 1, x, y, z, n);
printf("%d\n", ans);
}
else if(op == 3){
scanf("%d%d", &x, &y);
segment_tree_add(1, n, 1, x, y);
}
else if(op == 4){
scanf("%d%d%d", &x, &y, &z);
int k = segment_tree_query_lessk(1, n, 1, x, y, z);
k -= segment_tree_find(1, n, 1, x, y, z);
int ans = segment_tree_query_NO_k(1, n, 1, x, y, k, n);
printf("%d\n", ans);
}
else{
scanf("%d%d%d", &x, &y, &z);
int k = segment_tree_query_lessk(1, n, 1, x, y, z);
k ++;
int ans = segment_tree_query_NO_k(1, n, 1, x, y, k, n);
printf("%d\n", ans);
}
}
return 0;
}
BZOJ3196——二逼平衡树的更多相关文章
- BZOJ3196二逼平衡树——线段树套平衡树(treap)
此为平衡树系列最后一道:二逼平衡树您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名2.查询区间内排名为k的值3.修改某一位值上的数值4.查询 ...
- bzoj3196 二逼平衡树 树状数组套线段树
题目传送门 思路:树状数组套线段树模板题. 什么是树状数组套线段树,普通的树状数组每个点都是一个权值,而这里的树状数组每个点都是一颗权值线段树,我们用前缀差分的方法求得每个区间的各种信息, 其实关键就 ...
- bzoj3196 二逼平衡树 树套树(线段树套Treap)
Tyvj 1730 二逼平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 4697 Solved: 1798[Submit][Status][D ...
- bzoj3196 二逼平衡树
题目链接 平衡树系列最后一题 坑啊 10s时间限制跑了9764ms...还是要学一学bit套主席树啦... 经典的线段树套treap...至于第一发为什么要TLE(我不会告诉你treap插入的时候忘了 ...
- BZOJ3196 二逼平衡树 ZKW线段树套vector(滑稽)
我实在是不想再打一遍树状数组套替罪羊树了... 然后在普通平衡树瞎逛的时候找到了以前看过vector题解 于是我想:为啥不把平衡树换成vector呢??? 然后我又去学了一下ZKW线段树 就用ZKW线 ...
- [BZOJ3196] 二逼平衡树 [权值线段树套位置平衡树]
题面 洛咕题面 思路 没错我就是要不走寻常路! 看看那些外层位置数据结构,必须二分的,$O(n\log^3 n)$的做法吧! 看看那些cdq分治/树状数组套线段树的,空间$O(n\log^2 n)$挤 ...
- BZOJ3196 二逼平衡树 【线段树套平衡树】
题目 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的数值 4.查询k在区间内的前驱(前驱 ...
- luogu3380/bzoj3196 二逼平衡树 (树状数组套权值线段树)
带修改区间K大值 这题有很多做法,我的做法是树状数组套权值线段树,修改查询的时候都是按着树状数组的规则找出那log(n)个线段树根,然后一起往下做 时空都是$O(nlog^2n)$的(如果离散化了的话 ...
- BZOJ3196:二逼平衡树(线段树套Splay)
Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的数值 4.查询k在 ...
随机推荐
- Docker入门教程(二)命令
Docker入门教程(二)命令 [编者的话]DockerOne组织翻译了Flux7的Docker入门教程,本文是系列入门教程的第二篇,介绍了Docker的基本命令以及命令的用法和功能. 在Docker ...
- 【原】jquery图片预览
平时我们在做图片上传的时候,如果可以让用户选择图片的时候,看到图片的效果,那这样用户体验会好很多,因为用户可以就可以决定是否继续用这张图片,尤其是和ajaxuploadfile结合使用的时候,图片的预 ...
- Jquery 实现密码框的显示与隐藏【转载自http://blog.csdn.net/fengzhishangsky/article/details/11809069】
<html> <head> <script type="text/JavaScript" src="jQuery-1.5.1.min.j ...
- this.name=name;和this.setName(name);的区别
其实一般属性设置为private后,才会写属性的set和get方法 在本类中可以用 this.name=name 但是,在其他类中药给name赋值,就只能用set了
- Android学习笔记——ProgressBarHandler
该工程的功能是实现点击按钮进度条按10%递增,使用的方式是Handler 以下的代码是MainActivity.java中的代码 package com.example.progressbarhand ...
- pulltorefresh滚动到底部
如果用ListView,让它滚动到顶部,一般是这样写的: if (!listView.isStackFromBottom()) { listView.setStackFromBottom(true); ...
- Behavior Trees
https://en.wikipedia.org/wiki/Behavior_Trees_(artificial_intelligence,_robotics_and_control) http:// ...
- css 网站变灰色
网站变灰色 html{ -webkit-filter: grayscale(%); -webkit-filter: grayscale(); filter: grayscale(%); filter: ...
- Senparc.Weixin.MP.Sample 配置redis服务器密码
redis.windows-service.conf中加 requirepass 你的密码 <!-- Cache.Redis连接配置 --> <add key="Cach ...
- python版本升级及pip部署方法
Python版本升级 CentOS 6.3自带的Python版本为2.6,首先需要升级到2.7版本.由于旧版本的Python已被深度依赖,所以不能卸载原有的Python,只能全新安装. 1.下载Pyt ...