题面

传送门

思路

这道题目是陈立杰论文《重量平衡树和后缀平衡树在信息学奥赛中的应用 》中关于重量平衡树维护序列排名算法的一个应用

具体方法为:令根节点保存一个实数区间$[0,1]$

若当前节点是$[l,r]$,则左儿子是$[l,mid]$,右儿子是$[mid,r]$

令$f[x]=(l[x]+r[x])*0.5$,则$f$比较大小则等价于序列中的元素比较大小

此时,用重量平衡树来实现这一过程,可以做到均摊$O(\log n)$修改,$O(1)$查询两个点之间的大小关系

本题中,考虑使用一棵平衡树来维护所有位置的排名,再用一棵权值线段树统计答案,权值线段树中保存对应区间的最大的点的下标,利用上面求出的$f$值来比大小

平衡树选取替罪羊树,实现时我们维护一个$node$类,保存它的左半和右半部分(本题中数是由两个部分组成的,也就是括号序列)在平衡树中的排名

插入时按照题目给定的方式比较即可,不需要删除

Code

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#include<cassert>
#include<vector>
using namespace std;
inline int read(){
int re=0,flag=1;char ch=getchar();
while(!isdigit(ch)){
if(ch=='-') flag=-1;
ch=getchar();
}
while(isdigit(ch)) re=(re<<1)+(re<<3)+ch-'0',ch=getchar();
return re*flag;
}
//scapegoat
double f[600010];int root,unb;
struct node{
int l,r;
friend bool operator >(node x,node y){
if(f[x.l]>f[y.l]) return 1;
if(f[x.l]==f[y.l]&&f[x.r]>f[y.r]) return 1;
return 0;
}
friend bool operator ==(node x,node y){
return (f[x.l]==f[y.l])&&(f[x.r]==f[y.r]);
}
};
namespace scp{
node a[600010];
int siz[600010],key[600010][2],ch[600010][2],pos[600010],cntn;
void update(int x){
siz[x]=siz[ch[x][0]]+siz[ch[x][1]]+1;
}
vector<int>str;
void dfs(int u){
if(!u) return;
dfs(ch[u][0]);
str.push_back(u);
dfs(ch[u][1]);
}
void build(int &cur,int l,int r,double ql,double qr){
if(l>r){cur=0;return;}
int mid=(l+r)>>1;double md=(ql+qr)*0.5;
cur=str[mid];
f[cur]=md;
build(ch[cur][0],l,mid-1,ql,md);
build(ch[cur][1],mid+1,r,md,qr);
update(cur);
}
void rebuild(int &u,double l,double r){
str.clear();
dfs(u);
build(u,0,str.size()-1,l,r);
}
double alpha=0.75;int cho;
int insert(int &cur,node val,double ql,double qr){
double mid=(ql+qr)*0.5;
if(!cur){
cur=++cntn;f[cur]=mid;a[cur]=val;siz[cur]=1;
return cur;
}
int p;
if(val==a[cur]){p=cur;return p;}
else{
siz[cur]++;
if(val>a[cur]) p=insert(ch[cur][1],val,mid,qr);
else p=insert(ch[cur][0],val,ql,mid);
}
if((double)(siz[cur]*0.75)>(double)(max(siz[ch[cur][0]],siz[ch[cur][1]]))){
if(unb){
if(ch[cur][0]==unb) rebuild(ch[cur][0],ql,mid);
else rebuild(ch[cur][1],mid,qr);
unb=0;
}
}
else unb=cur;
return p;
}
}
int maxn[1000010],pos[1000010];
void change(int l,int r,int p,int cur){
int mid=(l+r)>>1;
if(l==r){maxn[p]=l;return;}
if(mid>=cur) change(l,mid,p<<1,cur);
else change(mid+1,r,p<<1|1,cur);
if(f[pos[maxn[p<<1]]]>=f[pos[maxn[p<<1|1]]]) maxn[p]=maxn[p<<1];
else maxn[p]=maxn[p<<1|1];
}
int query(int l,int r,int ql,int qr,int num){
if(l>=ql&&r<=qr) return maxn[num];
int mid=(l+r)>>1,re,re1;
if(mid>=qr) return query(l,mid,ql,qr,num<<1);
else if(mid<ql) return query(mid+1,r,ql,qr,num<<1|1);
else{
re=query(l,mid,ql,mid,num<<1);
re1=query(mid+1,r,mid+1,qr,num<<1|1);
if(f[pos[re]]>=f[pos[re1]]) return re;
else return re1;
}
}
int n,Q;
int main(){
n=read();Q=read();int i,t1,t2,t3;char s[10];
scp::insert(root,(node){0,0},0,1);f[0]=-1;
for(i=1;i<=n;i++) pos[i]=1;
for(i=1;i<=n;i++) change(1,n,1,i);
while(Q--){
scanf("%s",s);
t1=read();t2=read();
if(s[0]=='C'){
t3=read();unb=0;
pos[t3]=scp::insert(root,(node){pos[t1],pos[t2]},0,1);
if(unb) scp::rebuild(root,0,1);
change(1,n,1,t3);
}
else printf("%d\n",query(1,n,t1,t2,1));
}
}

最新文章

  1. vue 配置文件详解
  2. OData 带更新的实例,并能取得元数据格式类型
  3. 【PHP面向对象(OOP)编程入门教程】16.__toString()方法
  4. WinForm 实现验证码
  5. ODAC with Oracle Developer Tools for Visual Studio
  6. Robotium双client測试框架
  7. C语言中scanf/fscanf 的%[]和%n说明符的使用方法
  8. js判断MAC地址
  9. IOS使用pods初次加载出现Pods-resources.sh: Permission denied错误的解决方案
  10. yolov3和darknet opencv版编译安装及基本测试
  11. 巧妙使用excel 实现行转列
  12. 最大子矩阵和问题dp
  13. class的真相
  14. [转帖]LCD与LED的区别之背光原理与优缺点对比介绍
  15. js form表单 鼠标移入弹出提示功能
  16. MyBatis初探
  17. C++中虚函数的作用
  18. spring boot项目,application.properties配置文件下中文乱码解决方案
  19. s5-10 路由
  20. jenkins+maven+svn构建项目,及远程部署war包到tomcat上

热门文章

  1. 洛谷P1762 偶数(找规律)
  2. Linux实战教学笔记16:磁盘原理
  3. [异常笔记] spring cloud 服务消费者启动-2018040501
  4. Pythond函数的参数使用操作注意事项
  5. Python学习之函数参数
  6. C语言结构体篇 结构体
  7. 中国剩余定理算法详解 + POJ 1006 Biorhythms 生理周期
  8. EF实体部分更新的问题
  9. python字典的整理信息
  10. [bzoj2932][POI1999]树的染色问题