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. iOS安全攻防之阻止GDB依附

    GDB是大多数hackers的首选,阻止GDB依附到应用的常规办法是: #import <dlfcn.h> #import <sys/types.h> typedef int ...

  2. Docker - 运行 containers 使用在 swarm 模式下创建的 overlay 模式的 network

    前言 在Docker engine v1.12, 使用Swarm可以方便的创建overlay模式的网络,但是它只能被swarm下面的service所使用的,相对于container,这个网络是完全隔离 ...

  3. Redis 学习之简介及安装

    一.redis简介 Redis是一个开源的,先进的key-value存储.它通常被称为数据结构服务器,因为键可以包含字符串.哈希.链表.集合和有序集合. 支持的数据类型:string(字符串).lis ...

  4. angularJS ng-change错误的解决方案

    导入文件:<script src="../../js/angular/angular-file-upload/angular-file-upload.js"></ ...

  5. std::cin>>

    cin>> 不吃最后的回车换行,字符串自动补'\0'与最后回车换行无关 时常忘记,紧记!

  6. javascript常用开发笔记:一个简单强大的js日期格式化方法

    前言:一直没找到好用的javascript日期格式化工具,自己写了一个,方便以后复用 1.主要功能 (1)支持任意顺序的日期格式排列:yyyy=年,MM=月,dd=日,HH=时,mm=分,ss=秒,s ...

  7. Java开发中的23+2种设计模式学习个人笔记(未完待续)

    注:个人笔记 一.设计模式分三大类: 创建型模式,共五种:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式. 结构型模式,共七种:适配器模式.装饰器模式.代理模式.外观模式.桥接模式.组合模 ...

  8. python之路第一篇

    一.python环境的搭建 1.window下环境的搭建 (1).在 https://www.python.org/downloads/ 下载自己系统所需要的python版本 (2).安装python ...

  9. docker疑难解答 -- 设置远程服务监听

    环境: ubuntu 16.04 tls docker version 17.05.0-ce ========================= 今天想要搭建一个多主机的集成docker环境,但是我最 ...

  10. javascript走马灯的效果(文档标题文字滚动)

    做一些网站的时候,文档标题会滚动,这个效果是走马灯的效果. <!DOCTYPE html> <html> <head> <meta charset=" ...