方伯伯的OJ

题目描述

方伯伯正在做他的OJ。现在他在处理OJ 上的用户排名问题。

OJ 上注册了个用户,编号为1  n,一开始他们按照编号排名。方伯伯会按照心情对这些用户做以下四种操作,修改用户的排名和编号:

1. 操作格式为1 x y,意味着将编号为的用户编号改为y,而排名不变,执行完该操作后需要输出该用户在队列中的位置,数据保证必然出现在队列中, 同时是一个当前不在排名中的编号。

2. 操作格式为2 x,意味着将编号为的用户的排名提升到第一位,执行完该操作后需要输出执行该操作前编号为用户的排名。

3. 操作格式为3 x,意味着将编号为的用户的排名降到最后一位,执行完该操作后需要输出执行该操作前编号为用户的排名。

4. 操作格式为4 k,意味着查询当前排名为的用户编号,执行完该操作后需要输出当前操作用户的编号。

但同时为了防止别人监听自己的工作,方伯伯对他的操作进行了加密,即将四种操作的格式分别改为了:

a y a

a

a

a

其中为上一次操作得到的输出,一开始= 0。

例如:

上一次操作得到的输出是5

这一次操作的输入为:1 13 15

因为这个输入是经过加密后的,所以你应该处理的操作是1 8 10

现在你截获了方伯伯的所有操作,希望你能给出结果。

输入格式

输入的第1 行包含2 个用空格分隔的整数m,表示初始用户数和操作数。

此后有行,每行是一个询问,询问格式如上所示。

输出格式

输出包含行。每行包含一个整数,其中第行的整数表示第个操作的输出。

样例

样例输入

10 10
1 2 11
3 13
2 5
3 7
2 8
2 10
2 11
3 14
2 18
4 9

样例输出

2
2
2
4
3
5
5
7
8
11

数据范围与提示

【数据范围】
对于10% 的数据,1 ≤ n ≤ 10^3
对于40% 的数据,1 ≤ n ≤ 10^5
对于100% 的数据,1 ≤ n ≤ 10^8,1 ≤ m ≤ 105
输入保证对于所有的操作1,2,3,x 必然已经出现在队列中,同时对于所有操作1,1 ≤ y ≤ 2 ×10^8,并且y 没有出现在队列中。对于所有操作4,保证1 ≤ k ≤ n。

solution
可以发现1e8的数据啥也开不起来。
我们考虑用 splay 维护一段区间,这样的状态数是m了。
还是以splay的大小关系表示排名,节点维护区间长度。
对于1操作,我们用两个map将编号对应到 1~1e8
对于2.3操作,我们找到包含这个点的区间。然后把它拆成三块。
对于4操作,splay上找k大即可。
现在还剩一个问题,怎么找到包含某个点的区间。
邵神犇:用一个set把当前所有区间的左端点存起来,每次找第一个包含它的。(%%%
那就解决啦。
 
然而这题全是细节。。。
1.找第k大时找到了也要减左儿子
if(tr[ls].sz>=x)k=ls;
else if(tr[ls].sz+tr[k].r-tr[k].l+>=x){x-=tr[ls].sz;break;}
else x-=tr[ls].sz+tr[k].r-tr[k].l+,k=tr[k].ch[];

2.insert完要维护一整条链。

3.区间的size为r-l+1

4.根的前驱不是左儿子!!根的后驱不是右儿子!!(可能是我傻逼)

5.注意分讨x=l x=r x=l=r的特殊情况

6.节点开2m~3m

#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<map>
#include<set>
using namespace std;
int n,m,rt,ans,cnt;
struct node{
int sz,l,r,ch[],f;
}tr[];
set<int>s;
map<int,int>ls,dy,id;
void print_a(int k){
if(!k)return;
print_a(tr[k].ch[]);
for(int i=tr[k].l;i<=tr[k].r;i++)printf("%d ",i);
print_a(tr[k].ch[]);
}
bool get(int x){return tr[tr[x].f].ch[]==x;}
void wh(int x){
tr[x].sz=tr[tr[x].ch[]].sz+tr[tr[x].ch[]].sz+tr[x].r-tr[x].l+;
}
void rotate(int x){
int y=tr[x].f,z=tr[y].f;
int wx=get(x),wy=get(y);
tr[z].ch[wy]=x;tr[x].f=z;
tr[y].ch[wx]=tr[x].ch[wx^];tr[tr[x].ch[wx^]].f=y;
tr[x].ch[wx^]=y;tr[y].f=x;
wh(y);wh(x);
}
void splay(int x,int g){
while(tr[x].f!=g){
int y=tr[x].f,z=tr[y].f;
if(z!=g)rotate(get(x)==get(y)?y:x);
rotate(x);
}
if(!g)rt=x;
}
void ask(int t){
set<int>::iterator pos;
pos=s.upper_bound(t);pos--;
int k=id[*pos];
//cout<<"aaa "<<t<<' '<<*pos<<' '<<k<<' '<<tr[k].l<<' '<<tr[k].r <<endl;
splay(k,);
ans=tr[tr[rt].ch[]].sz;
ans+=t-(*pos)+;
printf("%d\n",ans);
}
void insert_Front(int x){
int k=rt;
while(tr[k].ch[])k=tr[k].ch[];
int p=++cnt;
tr[p].l=tr[p].r=x;tr[p].sz=;
tr[k].ch[]=p;tr[p].f=k;
for(int i=p;i;i=tr[i].f)wh(i);
s.insert(x);id[x]=cnt;
splay(p,);
}
void insert_Bottom(int x){
int k=rt;
while(tr[k].ch[])k=tr[k].ch[];
int p=++cnt;
tr[p].l=tr[p].r=x;tr[p].sz=;
tr[k].ch[]=p;tr[p].f=k;
for(int i=p;i;i=tr[i].f)wh(i);
s.insert(x);id[x]=cnt;
splay(p,);
//cout<<p<<' '<<tr[p].l<<' '<<tr[p].r<<' '<<tr[p].f<<endl;
}
void add(int x,int opt){
int l=tr[rt].l,r=tr[rt].r;
if(l==r){
int ls=tr[rt].ch[],rs=tr[rt].ch[];
if(!ls)rt=rs,tr[rs].f=;
else if(!rs)rt=ls,tr[ls].f=;
else {
while(tr[ls].ch[])ls=tr[ls].ch[];
while(tr[rs].ch[])rs=tr[rs].ch[];
splay(ls,);splay(rs,ls);
//print_a(rt);puts("");puts("------");
tr[tr[rt].ch[]].ch[]=;
wh(tr[rt].ch[]);wh(rt);
//print_a(rt);puts("");puts("------");
}
}
else if(x==l){ tr[rt].l=x+,wh(rt);
s.insert(x+);id[x+]=rt;
}
else if(x==r){
tr[rt].r=x-,wh(rt);
}
else {
int k=++cnt;
tr[k].l=x+;tr[k].r=tr[rt].r;
tr[rt].r=x-;
int rs=tr[rt].ch[];
if(rs){tr[k].ch[]=rs;tr[rs].f=k;}
tr[rt].ch[]=k;tr[k].f=rt;
wh(k);wh(rt);
//printf("id%d l=%d r=%d rs=%d k=%d\n",rt,tr[rt].l,tr[rt].r,tr[rt].ch[1],k);
//printf("kid%d kl=%d lr=%d\n",k,tr[k].l,tr[k].r);
s.insert(x+);id[x+]=k; }
if(!opt)insert_Front(x);
else insert_Bottom(x);
}
void Kth(int x){
int k=rt;
while(){
//printf("k:%d l=%d r=%d\n",k,tr[k].l,tr[k].r);
int ls=tr[k].ch[];
//printf("sz:%d\n",tr[ls].sz);
if(tr[ls].sz>=x)k=ls;
else if(tr[ls].sz+tr[k].r-tr[k].l+>=x){x-=tr[ls].sz;break;}
else x-=tr[ls].sz+tr[k].r-tr[k].l+,k=tr[k].ch[];
}
//cout<<tr[k].l<<' '<<x<<endl;
int f=tr[k].l+x-;
ans=dy[f];if(!ans)ans=f;
printf("%d\n",ans);
} int main()
{
freopen("test.in","r",stdin);
freopen("test.out","w",stdout);
cin>>n>>m;
rt=;cnt=;
tr[].l=,tr[].r=n,tr[].sz=n;
s.insert();id[]=;
ans=;
//print_a(rt);puts("");
for(int i=,op,x,y;i<=m;i++){
scanf("%d%d",&op,&x);x-=ans;
if(op==){
scanf("%d",&y);
y-=ans;
int t=ls[x];if(!t)t=x;
ls[y]=t;dy[t]=y;
ask(t);
}
if(op==){
int t=ls[x];if(!t)t=x;
ask(t);add(t,);
}
if(op==){
int t=ls[x];if(!t)t=x;
ask(t);add(t,);
}
if(op==){
Kth(x);
}
//cout<<"root "<<rt<<endl;
//print_a(rt);puts("");puts("------");
}
return ;
}
 

方伯伯的OJ ( onlinejudge )的更多相关文章

  1. SCOI2014 方伯伯的OJ onlinejudge

    终于用自己的方法水过去了. 本地测慢的一组要三四秒,一共要十几秒,BZOJ貌似一共只让跑6s,于是就还T着的. 一开始没看n<=1e8,想的直接splay+map(splay维护名次,map维护 ...

  2. BZOJ 3595: [Scoi2014]方伯伯的Oj SBT+可持久化Treap

    3595: [Scoi2014]方伯伯的Oj Time Limit: 6 Sec  Memory Limit: 256 MBSubmit: 102  Solved: 54[Submit][Status ...

  3. 「SCOI2014」方伯伯的 OJ 解题报告

    「SCOI2014」方伯伯的 OJ 和列队有点像,平衡树点分裂维护即可 但是需要额外用个set之类的对编号查找点的位置 插入完了后记得splay,删除时注意特判好多东西 Code: #include ...

  4. luogu P3285 [SCOI2014]方伯伯的OJ splay 线段树

    LINK:方伯伯的OJ 一道稍有质量的线段树题目.不写LCT splay这辈子是不会单独写的 真的! 喜闻乐见的是 题目迷惑选手 \(op==1\) 查改用户在序列中的位置 题目压根没说位置啊 只有排 ...

  5. 洛谷P3285 [SCOI2014]方伯伯的OJ 动态开点平衡树

    洛谷P3285 [SCOI2014]方伯伯的OJ 动态开点平衡树 题目描述 方伯伯正在做他的 \(Oj\) .现在他在处理 \(Oj\) 上的用户排名问题. \(Oj\) 上注册了 \(n\) 个用户 ...

  6. [SCOI2014]方伯伯的OJ(线段树)

    方伯伯正在做他的Oj.现在他在处理Oj上的用户排名问题.Oj上注册了n个用户,编号为1-n“,一开始他们按照编号排名. 方伯伯会按照心情对这些用户做以下四种操作,修改用户的排名和编号: 1.操作格式为 ...

  7. [SCOI2014]方伯伯的OJ

    看到这道题的第一想法就是要用FHQ treap 过了这道题...于是至今尚未成功(华丽的 T 掉了 (╯‵□′)╯︵┻━┻ ).于是附个地址. 然后水一波博客. 题意简介 emmmm...方伯伯脑抽做 ...

  8. 洛谷 P3285 / loj 2212 [SCOI2014] 方伯伯的 OJ 题解【平衡树】【线段树】

    平衡树分裂钛好玩辣! 题目描述 方伯伯正在做他的 OJ.现在他在处理 OJ 上的用户排名问题. OJ 上注册了 \(n\) 个用户,编号为 \(1\sim n\),一开始他们按照编号排名.方伯伯会按照 ...

  9. BZOJ 3595: [Scoi2014]方伯伯的Oj Splay + 动态裂点 + 卡常

    Description 方伯伯正在做他的Oj.现在他在处理Oj上的用户排名问题. Oj上注册了n个用户,编号为1-”,一开始他们按照编号排名.方伯伯会按照心情对这些用户做以下四种操作,修改用户的排名和 ...

随机推荐

  1. CSU1216: 异或最大值(01Trie树)

    Description 给定一些数,求这些数中两个数的异或值最大的那个值 Input 多组数据.第一行为数字个数n,1 <= n <= 10 ^ 5.接下来n行每行一个32位有符号非负整数 ...

  2. datatable常用设置

    bSort: false, // 是否排序功能 bFilter: false, // 过滤功能 bPaginate: true, // 翻页功能 bInfo: true, // 页脚信息 bProce ...

  3. zeppelin ERROR总结

    ERROR [2017-03-23 20:01:50,799] ({qtp331657670-221} NotebookServer.java[onMessage]:221) - Can't hand ...

  4. 记一次微信小程序在安卓的白屏问题

    在做小程序的时候,做到了一个限时商品售卖,用到了倒计时,因为这个原因导致了安卓手机上使用小程序时,将小程序放入后台运行一段时间后,再次进入小程序后出现了页面白屏或者点击事件失效的情况,这里记录下 1. ...

  5. Drazil and Tiles CodeForces - 516B (类拓扑)

    Drazil created a following problem about putting 1 × 2 tiles into an n × m grid: "There is a gr ...

  6. Oozie 实战之 Hive

    1.编辑job.propertiers nameNode=hdfs://cen-ubuntu.cenzhongman.com:8020 jobTracker=localhost:8032 queueN ...

  7. 【Noise and Probabilistic Target】林轩田机器学习基石

    http://beader.me/mlnotebook/section2/noise-and-error.html 上面这个日志总结的已经很好了.这一章的内容,在后面具体的算法中cost functi ...

  8. java中多态的概念

    概念: 简答来说,多态(Polymorphism)是具有表现多种形态的能力的特征.    更专业的说法是:同一个实现接口,使用不同的实例而执行不同的操作. 好处: 通过多态可以减少类中代码量,可以提高 ...

  9. CS/BS架构的特点

    CS架构 优点: 1.有独立的客户端,安全性高 2.大部分业务都在客户端实现,可以实现很复杂的业务 缺点: 1.对环境要求高,需要安装客户端,推广速度慢 2.需要专门前后台的开发团队,维护成本高 B/ ...

  10. Springboot 启动问题

    每次以debug方式启动springboot之后都会在SilentExitExceptionHandler类中的throw new SilentExitException() 解决办法 :window ...