平衡树蒟蒻,敲了半天。

其实思路很简单,就是把许多个人合并成一个区间。必要的时候再拆开。(是不是和这个题的动态开点线段树有异曲同工之妙?)

每次操作最多多出来6个点。

理论上时间复杂度是nlogn,空间复杂度实际远远小于上界,其实4*n即可。

出错点:

1.pushup把哨兵sz变成了1:pushup要特判x!=0(比较保险)

2.fa和son的边一定要成对一起添加!!t[t[y].ch[0]=t[x].ch[0]].fa=y 压行的话不容易漏。

3.对于一列的情况,可能删除之后整个树空了。但是如果直接找的话,因为哨兵0有奇奇怪怪的左右儿子和father,所以,一路上会把0的sz变成1,并且访问到奇奇怪怪的点23333

那就特判如果没有根的话,那就直接建立即可。

代码:

#include<bits/stdc++.h>
#define reg register int
#define il inline
using namespace std;
typedef long long ll;
const int N=+;
struct node{
int ch[];
int sz;
int L,R;
int fa;
ll val;
}t[N*];
int tot;
void rd(int &x){
char ch;x=;
while(!isdigit(ch=getchar()));
for(x=ch^'';isdigit(ch=getchar());x=x*+(ch^''));
}
int rt[N];
int now;
int n,m,q;
void pushup(int x){
if(x)
t[x].sz=t[t[x].ch[]].sz+t[t[x].ch[]].sz+t[x].R-t[x].L+;
}
void rotate(int x){
int y=t[x].fa,d=(t[y].ch[]==x);
t[t[y].ch[d]=t[x].ch[!d]].fa=y;
t[t[x].fa=t[y].fa].ch[t[t[y].fa].ch[]==y]=x;
t[t[x].ch[!d]=y].fa=x;
pushup(y);
}
void splay(int x,int f){
while(t[x].fa!=f){
int y=t[x].fa,z=t[y].fa;
if(z!=f){
rotate((t[y].ch[]==x)==(t[z].ch[]==y)?y:x);
}
rotate(x);
}
pushup(x);
if(f==) rt[now]=x;
}
ll get(int pos){
if(now==n+){
return (ll)pos*m;
}
return (ll)(now-)*m+pos;
}
void ins(int x,ll c){
if(x==){
++tot;
t[tot].fa=;
t[tot].val=c;
t[tot].L=t[tot].R=;
pushup(tot);
rt[now]=tot;
}
else{
while(t[x].ch[]) t[x].sz++,x=t[x].ch[]; ++tot;
t[x].ch[]=tot;
t[tot].fa=x;
t[tot].L=t[tot].R=;t[tot].val=c;
t[tot].sz=;
pushup(x);
splay(tot,);
int tmp=rt[now];
}
}
ll query(int x,int k){//and dele
int d=k-t[t[x].ch[]].sz;
if(d<=) return query(t[x].ch[],k);
else if(t[x].R-t[x].L+<d) return query(t[x].ch[],d-(t[x].R-t[x].L+));
else{//find position
ll ret=;
if(t[x].R>t[x].L){
if(d>){
++tot;
t[tot].L=t[x].L;t[tot].R=t[x].L+d-;
t[t[tot].ch[]=t[x].ch[]].fa=tot;
t[t[x].ch[]=tot].fa=x;
pushup(tot);
}
if(d<t[x].R-t[x].L+){
++tot;
t[tot].L=t[x].L+d;t[tot].R=t[x].R;
t[t[tot].ch[]=t[x].ch[]].fa=tot;
t[t[x].ch[]=tot].fa=x;
pushup(tot);
}
t[x].val=get(t[x].L+d-);
t[x].L=t[x].R=;
}else if(t[x].R){
t[x].val=get(t[x].R);
t[x].L=t[x].R=;
}
pushup(x); ret=t[x].val; splay(x,);
int son=t[x].ch[];
if(!son){
t[rt[now]=t[x].ch[]].fa=;
pushup(rt[now]);
}
else{
while(t[son].ch[]) son=t[son].ch[];
splay(son,rt[now]); t[t[son].ch[]=t[x].ch[]].fa=son;
t[son].fa=;
pushup(son);
rt[now]=son;
}
return ret;
}
}
int main(){
rd(n);rd(m);rd(q);
int x,y;
for(reg i=;i<=n;++i){
rt[i]=++tot;
t[tot].L=,t[tot].R=m-;
t[tot].sz=m-;
t[tot].fa=;
}
rt[n+]=++tot;
t[tot].L=,t[tot].R=n;
t[tot].sz=n;
while(q--){
rd(x);rd(y);
ll id=;
if(y==m) now=n+,id=query(rt[now],x);
else now=x,id=query(rt[now],y);
printf("%lld\n",id);
now=n+;
ins(rt[now],id);
if(y!=m){
now=n+;
id=query(rt[now],x);
now=x;
ins(rt[now],id);
}
}
return ;
}

NOIP2017 列队——平衡树的更多相关文章

  1. [UOJ#334][NOIP2017]列队 平衡树/线段树/树状数组

    题目链接 题意不说了,一辈子也忘不掉 解法1.平衡树 这题就是平衡树裸题,每一行开一棵维护前 \(m-1\) 个,最后一列单独维护,因为很多人没有用到,所以平衡树每个节点是一个区间(pair),分裂时 ...

  2. [NOIP2017]列队 离线+SBT

    [NOIP2017]列队 题目描述 Sylvia 是一个热爱学习的女♂孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有n×m名学生,方阵 ...

  3. 题解[NOIP2017] 列队

    题解[NOIP2017] 列队 题面 解析 看到这题时感觉这个编号很难维护啊? 后来看了lzf大佬的题解才会.. 首先,考虑一个稍微暴力的做法, 维护每一行的前\(m-1\)个人和最后一列的\(n\) ...

  4. [NOIP2017] 列队(平衡树)

    考虑转化题意: 设这次操作删掉点\((x, y)\) 对于每一次向左看齐:在第x行删除\((x, y)\),并将y以后的点全部前移一位 对于每一次向前看齐:x以后的点全部上移一位,并在最后一列插入\( ...

  5. [NOIP2017]列队 (Splay)

    题目链接 NOIP2017真的是不按常理出牌: 1.数学题不在Day2T1 2.一道水题一道细节极多的模拟题一道不知道怎么形容的题(小凯的疑惑)(因为我太菜了) 3.3道大火题 当时看到列队这题是毫无 ...

  6. NOIP2017列队(phalanx)解题报告

    列队作为NOIP2017最后一道题,其实并不难,只是相对于其它题目,有点小小的工业 首先,这道题我用splay维护的,如果你不会splay,又想学一下splay,可以来这里学一学,接下来步入正题 首先 ...

  7. NOIP2017 列队 题解报告【56行线段树】

    题目描述 Sylvia 是一个热爱学习的女♂孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有n \times mn×m名学生,方阵的行数 ...

  8. NOIP2017 列队——动态开点线段树

    Description: Sylvia 是一个热爱学习的女♂孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有n×m名学生,方阵的行数为  ...

  9. Luogu 3960 [NOIP2017] 列队 - splay|线段树

    题解 是我从来没有做过的裂点splay... 看的时候还是很懵逼的QAQ. 把最后一列的$n$个数放在一个平衡树中, 有 $n$ 个点 剩下的$n$行数, 每行都开一个平衡树,开始时每棵树中仅有$1$ ...

随机推荐

  1. 第六模块:WEB框架开发 第1章·Django框架开发88~128

    88-Ajax简介 89-Ajax的简单实现 90-基于Ajax的传递数据 91-基于Ajax的登录验证 92-基于Form表单的文件上传 93-请求头之contentType 94-Ajax传递js ...

  2. Java 语法基础

    一 关键字 关键字: 其实就是某种语言赋予了特殊含义的单词 保留字: 其实就是还没有赋予特殊含义 但是准备日后要使用过的单词 二 标示符 标示符: 其实就是在程序中自定义的名词 比如类名, 变量名, ...

  3. 【rich-text】 富文本组件说明

    [rich-text] 富文本组件可以显示HTML代码样式. 1)支持事件:tap.touchstart.touchmove.touchcancel.touchend和longtap 2)信任的HTM ...

  4. 浅谈CPU、内存、硬盘之间的关系

    计算机,大家都知道的,就是我们日常用的电脑,不管台式的还是笔记本都是计算机.那么这个看着很复杂的机器由哪些组成的呢,今天就简单的来了解一下. 先放图: 图上展示的就是计算机的基本组成啦. 首先是输入设 ...

  5. python3 bytes与hex_string之间的转换

    1, bytes to hex_string的转换: def byte_to_hex(bins): """ Convert a byte string to it's h ...

  6. instanceof 运算符简介

    文章摘自: http://www.ibm.com/developerworks/cn/web/1306_jiangjj_jsinstanceof/ https://developer.mozilla. ...

  7. 关于LNMP常见问题和性能方面的个人理解

    简单整理,自己做备忘的,不为其他作任何参考- PHP程序 1.开启慢日志,过滤超时时间为1s的方法,针对性优化,可以通过添加缓存方式解决. 2.过滤access日志,统计哪些请求较多较为频繁,是否合理 ...

  8. catalan卡塔兰数

    令h(0)=1,h(1)=1,卡塔兰数数满足递归式:h(n)= h(0)*h(n-1) + h(1)*h(n-2) + ... + h(n-1)h(0) (其中n>=2),这是n阶递推关系;还可 ...

  9. iOS- UITableViewCell对象是怎么重用的 ?

    iOS设备的内存有限,如果用UITableView显示成千上万条数据, 就需要成千上万个UITableViewCell对象的话, 那将会耗尽iOS设备的内存.要解决该问题,需要重用UITableVie ...

  10. iOS开发多线程编程2 - NSOperation

    1.简介 NSOperation实例封装了需要执行的操作和执行操作所需的数据,并且能够以并发或非并发的方式执行这个操作. NSOperation本身是抽象基类,因此必须使用它的子类,使用NSOpera ...