Problem 普通平衡树

Solution

本题是裸的二叉平衡树。有很多种方法可以实现。这里打的是替罪羊树模板。 
此题极其恶心。 
前驱后继模块需要利用到rank模块来换一种思路求。 
很多细节的地方容易炸。我拿数据调了很久才A。 
(delt()删除模块其实是不需要重建的,不影响时间复杂度) 
替罪羊树具体详见此篇知乎:替罪羊树

AC Code

 #include "iostream"
#include "cstdio"
#define alpha 0.7
using namespace std;
struct ScapeGoatTree{
int lc,rc,n,w,s,fa;
}a[];
struct REbuild{
int n,w;
}d[];
void rebuild(int p);
int tot=,tott=,top=,h[],n,T,X,root;
int check(int p){
if(a[p].w==)return ;
if ((a[a[p].lc].s+a[a[p].lc].w>=double(alpha*(a[p].w+a[p].s)))||
(a[a[p].rc].s+a[a[p].rc].w>=double(alpha*(a[p].w+a[p].s))))return ;
return ;
}
int getN(int x){
int now=root;
while(a[now].n!=x&&a[now].s!=){
if(a[now].n>=x)now=a[now].lc;
else now=a[now].rc;
}
return now;
}
void insr(int x,bool rb){
if(root==){
root=;
a[].n=x;
a[].w++;
return;
}
int now=root;
while(a[now].s!=){
if(x==a[now].n){
a[now].w++;
return;
}
if(x>a[now].n&&a[now].rc==)break;
if(x<a[now].n&&a[now].lc==)break;
a[now].s++;
if(x>a[now].n)now=a[now].rc;
else if(x<a[now].n)now=a[now].lc;
} if(x==a[now].n){
a[now].w++;
return;
}
a[now].s++;
int tmp=now;
if(x>a[now].n)now=a[now].rc=++tot;
else if(x<a[now].n)now=a[now].lc=++tot;
a[now].w++;
a[now].n=x;
if(tmp!=now)a[now].fa=tmp;
int chk=;
while(now!=root){
now=a[now].fa;
if(check(now))chk=now;
}
if(rb)rebuild(chk);
}
int getnewp(){
if(top>){
top--;
return h[top+];
}
else return ++tot;
}
void dfs(int p){
h[++top]=p;
if(a[p].lc!=)dfs(a[p].lc);
if(a[p].w!=)d[++tott].n=a[p].n,
d[tott].w=a[p].w;
a[p].s=;
if(a[p].rc!=)dfs(a[p].rc);
}
void make(int l,int r,int p,int fa){
int mid=(l+r)/;
a[p].fa=fa;
a[p].w=d[mid].w;
a[p].n=d[mid].n;
if(mid->=l)a[p].lc=getnewp(),make(l,mid-,a[p].lc,p);else a[p].lc=;
if(mid+<=r)a[p].rc=getnewp(),make(mid+,r,a[p].rc,p);else a[p].rc=;
a[p].s=a[a[p].lc].w+a[a[p].lc].s+a[a[p].rc].w+a[a[p].rc].s;
}
void rebuild(int p){
if(p==)return;
tott=;
dfs(p);
int now=getnewp();
a[now].fa=a[p].fa;
if(p==root)root=now;
int mid=(+tott)/;
if(d[mid].n>a[a[p].fa].n)a[a[p].fa].rc=now;
else a[a[p].fa].lc=now;
make(,tott,now,a[p].fa);
}
void delt(int p,bool rb){
a[p].w--;
int chk=;
while(p!=root){
p=a[p].fa;
a[p].s--;
if(check(p))chk=p;
}
// if(rb)rebuild(chk);
}
int XgetRk(int x);
int RkgetX(int x);
int suc(int x){
insr(x+,);
int tmp=XgetRk(x+),nx=getN(x+);
delt(nx,);
return RkgetX(tmp);
}
int pre(int x){
insr(x,);
int tmp=XgetRk(x);
delt(getN(x),);
return RkgetX(tmp-);
}
int XgetRk(int x){
int p=getN(x),ans;
ans=a[a[p].lc].s+a[a[p].lc].w;
while(p!=root){
if(a[a[p].fa].rc==p)ans+=a[a[a[p].fa].lc].s+a[a[a[p].fa].lc].w+a[a[p].fa].w;
p=a[p].fa;
}
return ans+;
}
int RkgetX(int x){
int now=root;
while(true){
int lcS=a[a[now].lc].s+a[a[now].lc].w;
if(x<=lcS)now=a[now].lc;
else if(x>lcS&&x<=a[now].w+lcS)return a[now].n;
else x-=lcS+a[now].w,now=a[now].rc;
if(x==)return a[now].fa;
}
}
int main(){
scanf("%d",&n);
for(int i=;i<=n;i++){
if(i%==){
int ttott;
ttott++;
}
scanf("%d%d",&T,&X);
if(T==)insr(X,);
else if(T==)delt(getN(X),);
else if(T==)printf("%d\n",XgetRk(X));
else if(T==)printf("%d\n",RkgetX(X));
else if(T==)printf("%d\n",pre(X));
else if(T==)printf("%d\n",suc(X));
}
}

[TYVJ1728/BZOJ3224]普通平衡树-替罪羊树的更多相关文章

  1. Luogu 3369 / BZOJ 3224 - 普通平衡树 - [替罪羊树]

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=3224 https://www.luogu.org/problemnew/show/P3 ...

  2. 平衡树 替罪羊树(Scapegoat Tree)

    替罪羊树(Scapegoat Tree) 入门模板题 洛谷oj P3369 题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作: 插入xx数 删除xx数(若有多个相同 ...

  3. bzoj2827: 千山鸟飞绝 平衡树 替罪羊树 蜜汁标记

    这道题首先可以看出坐标没有什么意义离散掉就好了. 然后你就会发现你要每次都更改坐标,而一旦更改受影响的是坐标里的所有数,要是一个一个的改,会不可描述. 所以换个视角,我们要找的是某只鸟所到每个坐标时遇 ...

  4. bzoj 3224: Tyvj 1728 普通平衡树 替罪羊树

    题目链接 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个)3. 查询x数的排名(若有多个相同的数,因输出最小的 ...

  5. [luogu3369]普通平衡树(替罪羊树模板)

    解题关键:由于需要根据平衡进行重建,所以不能进行去重,否则无法保证平衡性. #include<cstdio> #include<cstring> #include<alg ...

  6. Bzoj3224 / Tyvj 1728 普通替罪羊树

    Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 12015  Solved: 5136 Description 您需要写一种数据结构(可参考题目标题), ...

  7. 【替罪羊树】bzoj3224&luogu3369&cogs1829 [Tyvj 1728]普通平衡树

    [替罪羊树]bzoj3224&luogu3369&cogs1829 [Tyvj 1728]普通平衡树 bzoj 洛谷 cogs 先长点芝士 替罪羊树也是一种很好写的平衡树qwq..替罪 ...

  8. 替罪羊树—BZOJ3224: Tyvj 1728 普通平衡树

    冬令营被平衡树坑了之后,打算苦练一番数据结构(QAQ). 先是打了一下想学好久的替罪羊树. 替罪羊树实现方法很简单,就是在不满足平衡条件的时候暴力重构子树. 调试小结: 1.删除操作分两类情况:如果某 ...

  9. 平衡树简单教程及模板(splay, 替罪羊树, 非旋treap)

    原文链接https://www.cnblogs.com/zhouzhendong/p/Balanced-Binary-Tree.html 注意是简单教程,不是入门教程. splay 1. 旋转: 假设 ...

随机推荐

  1. jQuery-强大的jQuery选择器、过滤器

    1. 基础选择器 Basics 名称 说明 举例 #id 根据元素Id选择 $("divId") 选择ID为divId的元素 element 根据元素的名称选择, $(" ...

  2. 基于nodejs 的多页面爬虫

    前言 前端时间再回顾了一下node.js,于是顺势做了一个爬虫来加深自己对node的理解. 主要用的到是request,cheerio,async三个模块 request 用于请求地址和快速下载图片流 ...

  3. 分享一款在线less转css的神器

          大多数web开发的程序员都了解和使用过Less, LESS是一门 CSS 预处理语言,它扩充了 CSS 语言,增加了诸如变量.混合(mixin).函数等功能,让 CSS 更易维护.方便制作 ...

  4. poj1379

    poj1379 题意 给出 n 个洞的坐标,要求找到一点使得这一点距离最近洞的距离最远. 分析 通过这道题学习一下模拟退火算法, 这种随机化的算法,在求解距离且精度要求较小时很有用. 简而言之,由随机 ...

  5. Unity应用架构设计(11)——一个网络层的构建

    对于客户端应用程序,免不了和远程服务打交道.设计一个良好的『服务层』能帮我们规范和分离业务代码,提高生产效率.服务层最核心的模块一定是怎样发送请求,虽然Mono提供了很多C#网络请求类,诸如WebCl ...

  6. php注册登录源代码

    php注册登录源代码 链接数据库<?php$conn=mysql_connect('localhost','root','');mysql_select_db('ht',$conn);mysql ...

  7. form表单在前台转json对象

    会发生序列化乱码问题,待解决. //根据表单id将其内空间,名称,值转为json var fireTraceEquipment =queryParamByFormId('form1'); functi ...

  8. servlet与Javabean之间的区别

    在JSP中调用JAVA类和使用JavaBean有什么区别? 可以像使用一般的类一样使用JavaBean,Bean只是一种特殊的类.特殊在可以通过<jsp:useBean/>调用JavaBe ...

  9. solr5Ik分词2

    <!--IK分词器--><fieldType name="text_ik" class="solr.TextField"><ana ...

  10. shell脚本调用C语言之字符串切分之strtok函数

    今天上午在写一个需求,要求的比较急,要求当天完成,我大致分析了一下,可以采用从shell脚本中插入一连串的日期,通过调用proc生成的可执行文件,将日期传入后台数据库,在数据库中进行计算.需要切分日期 ...