BZOJ_3224 Tyvj 1728 普通平衡树 【离散化+权值线段树】
一 题面
二 分析
比较明显是可以用平衡二叉搜索树(splay)做的。
用权值线段树做,前提就是要先离散化,因为权值线段树维护的值域信息。
板子。
三 AC代码
#include <iostream>
#include <algorithm>
#include <cstring>
#include <fstream> using namespace std;
#define lson rt<<1
#define rson rt<<1|1 const int MAXN = 1e5 + ;
int N, op[MAXN], opx[MAXN];
struct Node
{
int id, x;
}Data[MAXN];
int A[MAXN]; int segTree[MAXN<<]; bool Cmpx(const Node &a, const Node &b)
{
return a.x < b.x;
} //p更新的位置,v=-1表示减,v=1表示加
void Update(int p, int v, int rt, int l, int r)
{
segTree[rt] += v;
if(l == r) return;
int mid = (l+r)>>;
if(p <= mid) Update(p, v, lson, l, mid);
else Update(p, v, rson, mid+, r);
}
//第K小
int Kth(int K, int rt, int l, int r)
{
if(l == r)
return l;
int mid = (l+r)>>;
if(segTree[lson] >= K)
return Kth(K, lson, l, mid);
else
return Kth(K-segTree[lson], rson, mid+, r);
}
//数字v的排名
//统计小于v的数的个数
int Rank(int v, int rt, int l, int r)
{
if(r < v)
return segTree[rt];
int mid = (l+r)>>, res = ;
res += Rank(v, lson, l, mid);
//相当于mid+1 < v 那么左边可能还有
if(mid < v-)
res += Rank(v, rson, mid+, r);
return res;
}
//找最右边的数
int Findr(int rt, int l, int r)
{
if(l == r)
return l;
int mid = (l+r)>>;
if(segTree[rson]) return Findr(rson, mid + , r);
return Findr(lson, l, mid);
}
//前驱
int Pre(int v, int rt, int l, int r)
{
if(r < v)
{
if(segTree[rt]) return Findr(rt, l, r);
return ;
}
int mid = (l+r)>>, res;
//如果v > mid+1那么意味着mid+1或后面的数有可能是前驱
if(mid < v- && segTree[rson] && (res=Pre(v,rson,mid+,r)))
return res;
return Pre(v, lson, l, mid);
}
//找最左边的数
int Findl(int rt, int l, int r)
{
if(l == r)
return l;
int mid = (l+r)>>;
if(segTree[lson]) return Findl(lson, l, mid);
return Findl(rson, mid+, r);
}
//后继
int Nex(int v, int rt, int l, int r)
{
if(v < l)
{
if(segTree[rt]) return Findl(rt, l, r);
return ;
}
int mid = (l+r)>>, res;
//如果mid > v表示左子树的数可能是后继
if(v < mid && segTree[lson] && (res=Nex(v,lson,l,mid)))
return res;
return Nex(v, rson, mid + , r);
} int main()
{
//freopen("input.txt", "r", stdin);
scanf("%d", &N);
for(int i = ; i <= N; i++)
{
scanf("%d %d", &op[i], &Data[i].x);
Data[i].id = i;
}
//离散化预处理
sort(Data + , Data + N + , Cmpx);
int cnt = ;
A[cnt] = Data[].x;
opx[Data[].id] = cnt;
for(int i = ; i <= N; i++)
{
if(Data[i].x != Data[i - ].x)
{
cnt++;
}
A[cnt] = Data[i].x;
opx[Data[i].id] = cnt;
}
//建树
memset(segTree, , sizeof(segTree) );
for(int i = ; i <= N; i++)
{
switch(op[i])
{
case : Update(opx[i], , , , cnt); break;
case : Update(opx[i], -, , , cnt); break;
case : printf("%d\n", Rank(opx[i], , , cnt)+); break;
//这里注意第K个数也离散化了!!!
case : printf("%d\n", A[ Kth(A[opx[i]], , , cnt)] ); break;
case : printf("%d\n", A[ Pre(opx[i], , , cnt)] ); break;
case : printf("%d\n", A[ Nex(opx[i], , , cnt)] ); break;
}
} return ;
}
BZOJ_3224 Tyvj 1728 普通平衡树 【离散化+权值线段树】的更多相关文章
- R - Weak Pair HDU - 5877 离散化+权值线段树+dfs序 区间种类数
R - Weak Pair HDU - 5877 离散化+权值线段树 这个题目的初步想法,首先用dfs序建一颗树,然后判断对于每一个节点进行遍历,判断他的子节点和他相乘是不是小于等于k, 这么暴力的算 ...
- HDU 6609 离散化+权值线段树
题意 有一个长度为\(n\)的数组W; 对于每一个\(i\)(\(1<=i<=n\)),你可以选择中任意一些元素W[k] (\(1<=k<i\)),将他们的值改变为0,使得\( ...
- P3369 【模板】普通平衡树(权值线段树)
原来线段树还有这种操作(开成一个桶) 用区间维护在这个区间内元素的个数,离散化一下,居然能达到splay的效果 不仅码量大大减少,而且跑的飞快!!! 6种操作 200多ms 插入 xx 数 删除 x ...
- bzoj1588: [HNOI2002]营业额统计(权值线段树)
1588: [HNOI2002]营业额统计 Time Limit: 5 Sec Memory Limit: 162 MBSubmit: 16863 Solved: 6789[Submit][Sta ...
- cogs 1829. [Tyvj 1728]普通平衡树 权值线段树
1829. [Tyvj 1728]普通平衡树 ★★★ 输入文件:phs.in 输出文件:phs.out 简单对比时间限制:1 s 内存限制:1000 MB [题目描述] 您需要写一种数 ...
- 线段树(单标记+离散化+扫描线+双标记)+zkw线段树+权值线段树+主席树及一些例题
“队列进出图上的方向 线段树区间修改求出总量 可持久留下的迹象 我们 俯身欣赏” ----<膜你抄> 线段树很早就会写了,但一直没有总结,所以偶尔重写又会懵逼,所以还是要总结一下. ...
- luogu3380/bzoj3196 二逼平衡树 (树状数组套权值线段树)
带修改区间K大值 这题有很多做法,我的做法是树状数组套权值线段树,修改查询的时候都是按着树状数组的规则找出那log(n)个线段树根,然后一起往下做 时空都是$O(nlog^2n)$的(如果离散化了的话 ...
- [BZOJ3600] 没有人的算术 [重量平衡树+权值线段树]
题面 传送门 思路 这道题目是陈立杰论文<重量平衡树和后缀平衡树在信息学奥赛中的应用 >中关于重量平衡树维护序列排名算法的一个应用 具体方法为:令根节点保存一个实数区间$[0,1]$ 若当 ...
- [bzoj3196][Tyvj1730]二逼平衡树_树套树_位置线段树套非旋转Treap/树状数组套主席树/权值线段树套位置线段树
二逼平衡树 bzoj-3196 Tyvj-1730 题目大意:请写出一个维护序列的数据结构支持:查询给定权值排名:查询区间k小值:单点修改:查询区间内定值前驱:查询区间内定值后继. 注释:$1\le ...
随机推荐
- InstaGAN: Instance-Aware Image-to-Image Translation
- 通过HttpClient请求webService
通过HttpClient请求webService 由于服务端是用webService开发的,android要调用webService服务获取数据,这里采用的是通过HttpClient发送post请求, ...
- ASP.NET Core2读写InfluxDB时序数据库
在我们很多应用中会遇到有一种基于一系列时间的数据需要处理,通过时间的顺序可以将这些数据点连成线,再通过数据统计后可以做成多纬度的报表,也可通过机器学习来实现数据的预测告警.而时序数据库就是用于存放管理 ...
- visual studio code中使用emmet插件在.vue文件失效
使用visual studio code编辑.vue文件时,emmet插件无法使用,可以通过以下两种试解决: 1.文件→设置,在右侧窗口添加以下代码: "emmet.syntaxProfil ...
- GBK还是UTF-8? Eclipse连接TFS的编码之痛!encoding, encoding, encoding…
在中文Windows操作系统上安装Eclipse或MyEclipse,默认会将Eclipse的编码设置为GBK,与操作系统的默认编码保存一致. 在这种默认设置下,在Eclipse新增的文件不会自动被团 ...
- SelectOnCheck
1.checkOnSelect 如果为true,当用户点击行的时候该复选框就会被选中或取消选中. 如果为false,当用户仅在点击该复选框的时候才会呗选中或取消. 2.selectOnCheck 如果 ...
- .net core执行dotnet ef migrations createmodel等命令出错
.net core执行dotnet ef migrations createmodel等命令出错 执行dotnet ef migrations createmodel.dotnet ef migrat ...
- Polynomial ( Arithmetic and Algebra) CGAL 4.13 -User Manual
1 Fundamentals A polynomial is either zero, or can be written as the sum of one or more non-zero ter ...
- 「BZOJ1426」收集邮票
题目链接 戳我 \(Solution\) 我们首先转换一下问题: 假设我们进行了k轮得到了所有种类的邮票 则所花费用为: \[(1+2+5+...+k)=\frac{(1+k)*k}{2}=\frac ...
- ADV三星
#include <iostream> using namespace std; #define SIZE 12 int data[SIZE]; int data1[SIZE]; int ...