方伯伯正在做他的Oj。现在他在处理Oj上的用户排名问题。Oj上注册了n个用户,编号为1~n“,一开始他们按照编号排名。

方伯伯会按照心情对这些用户做以下四种操作,修改用户的排名和编号:

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

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

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

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

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

  • 1 x+a y+a
  • 2 x+a
  • 3 x+a
  • 4 k+a
  • 其中a为上一次操作得到的输出,一开始a=0。

例如:上一次操作得到的输出是5这一次操作的输入为:1 13 15因为这个输入是经过加密后的,所以你应该处理的操作是1 8 10现在你截获了方伯伯的所有操作,希望你能给出结果。

Solution

调了一晚上。。。菜的不行。

因为数列每次只会变化一个数,所以可以不用splay或fhqtreap,用一颗动态开点线段树就可以完成。

因为我们的移动只关于先插和后插,所以我们线段树的区间为m+n+m对于一个移动,我们先把该点在对应位置删除,再将该数插入前/后对应位置。

与此同时,我们开两个map,一个储存编号对应的位置,一个储存位置对应的编号。

Code

#include<iostream>
#include<cstdio>
#include<map>
#define ls tr[cnt].l
#define rs tr[cnt].r
#define N 200002
using namespace std;
map<int,int>mp,anti_mp;
int tot,id,ans,n,m,root,h,t,x,y;
struct seg{
int l,r,num;
bool la;
}tr[N*];
inline int rd(){
int x=;char c=getchar();
while(!isdigit(c))c=getchar();
while(isdigit(c)){
x=(x<<)+(x<<)+(c^);
c=getchar();
}
return x;
}
inline void pushdown(int cnt,int l1,int l2){
if(!ls)ls=++tot;if(!rs)rs=++tot;
tr[ls].num=l1;tr[rs].num=l2;
tr[cnt].la=;tr[ls].la=tr[rs].la=;
}
void add(int &cnt,int l,int r,int x,int tag,int rk){
if(!cnt)cnt=++tot;
if(l==r){
if(tag<)ans=rk+,anti_mp.erase(l);
else anti_mp[l]=id;
tr[cnt].num-=tag;
return;
}
int mid=(l+r)>>;
if(tr[cnt].la)pushdown(cnt,mid-l+,r-mid);
if(mid>=x)add(ls,l,mid,x,tag,rk);
else add(rs,mid+,r,x,tag,rk+mid-l+-tr[ls].num);
tr[cnt].num=tr[ls].num+tr[rs].num;
}
void find(int &cnt,int l,int r,int rk){
if(!cnt)cnt=++tot;
if(l==r){
if(anti_mp.find(l)!=anti_mp.end())ans=anti_mp[l];else ans=l-m;
return;
}
int mid=(l+r)>>;
if(tr[cnt].la)pushdown(cnt,mid-l+,r-mid);
int num=mid-l+-tr[ls].num;
if(num>=rk)find(ls,l,mid,rk);
else find(rs,mid+,r,rk-num);
}
void gai(int &cnt,int l,int r,int L,int R){
if(!cnt)cnt=++tot;
if(l>=L&&r<=R){
tr[cnt].num=r-l+;
tr[cnt].la=;
return;
}
int mid=(l+r)>>;
if(mid>=L)gai(ls,l,mid,L,R);
if(mid<R)gai(rs,mid+,r,L,R);
tr[cnt].num=tr[ls].num+tr[rs].num;
}
void upd(int &cnt,int l,int r,int x,int y,int rk){
if(!cnt)cnt=++tot;
if(l==r){
anti_mp[l]=y;ans=rk+;
return;
}
int mid=(l+r)>>;
if(tr[cnt].la)pushdown(cnt,mid-l+,r-mid);
if(mid>=x)upd(ls,l,mid,x,y,rk);
else upd(rs,mid+,r,x,y,rk+mid-l+-tr[ls].num);
}
int main(){
n=rd();m=rd();
gai(root,,n+*m,,m);
gai(root,,n+*m,n+m+,n+m*);
h=m+;t=n+m;
for(int i=;i<=m;++i){
int tag=rd();x=rd()-ans;
if(tag==){
y=rd()-ans;
int pos;
if(mp.find(x)!=mp.end())pos=mp[x];
else pos=x+m;
mp[y]=pos;
upd(root,,n+*m,pos,y,);
}
else if(tag==){
int pos;
if(mp.find(x)!=mp.end())pos=mp[x];
else pos=x+m;
if(anti_mp.find(pos)!=anti_mp.end())id=anti_mp[pos];
else id=x;
add(root,,n+*m,pos,-,);
add(root,,n+*m,--h,,);
mp[x]=h;
}
else if(tag==){
int pos;
if(mp.find(x)!=mp.end())pos=mp[x];
else pos=x+m;
if(anti_mp.find(pos)!=anti_mp.end())id=anti_mp[pos];
else id=x;
add(root,,n+*m,pos,-,);
add(root,,n+*m,++t,,);
mp[x]=t;
}
else find(root,,n+*m,x);
printf("%d\n",ans);
}
return ;
}

[SCOI2014]方伯伯的OJ(线段树)的更多相关文章

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

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

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

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

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

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

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

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

  5. [SCOI2014]方伯伯的OJ

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

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

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

  7. BZOJ3595 : [Scoi2014]方伯伯的Oj

    由于n很大,有2e8,所以不能直接用splay来维护排名 把splay修改一下 每个节点维护一个区间[l,r],表示编号在[l,r]之间的所有点都在这里 需要支持一个takeout操作: 把编号为k的 ...

  8. 洛谷 P3285 [SCOI2014]方伯伯的OJ

    看到这题,第一眼:平衡树水题,随便做一做好了 然后....我在花了n个小时去调试(维护平衡树父节点)之后,... 调了三个小时后,第一次失败的代码(只能查找排名为k的用户编号,不能根据编号查排名) # ...

  9. [Scoi2014]方伯伯的OJ(动态开点splay)

    开始没看数据范围差点以为是这题了:https://www.cnblogs.com/hfctf0210/p/10911340.html 然后看到n<=1e8,怎么这么大? 所以这题需要用动态开点线 ...

随机推荐

  1. mysql uuid() 相同 重复

    mysql select UPPER(REPLACE(uuid(),'-','')) from xxxtable 得到相同的uuid的问题 - LWJdear的博客 - CSDN博客 https:// ...

  2. 05 Hadoop 设置块的大小

    1.是在hdfs的配置文件中配置 2.是在app程序中设置 注意:假设配置文件的最大是   20K   最小是 10K   文件大小为72  块数就是 4 在程序中设置最大为15K    切割块数  ...

  3. [转帖]SAP一句话入门:Plant Maintenance

    SAP一句话入门:Plant Maintenance http://blog.vsharing.com/MilesForce/A618273.html PM就是Plant Maintenance(本文 ...

  4. <转>Python中的新式/经典类的查找方式

    在学习到深度和广度的时候,懵了很久.后来看到这篇文章,恍然大悟.写的很好.特意转过来. 经典类: 只要有父类, 就会沿着一直找, 即使已经找过了~ 新式类: 在类继承的多个类拥有共同父类的情况下, 会 ...

  5. Java基础——对象容器(顺序、集合、Hash)

    扩展: For-each循环 for (String s: str) { System.out.println(s); } 等同于for (int i = 0; i < str.length; ...

  6. mysql对身份证号码进行脱敏处理

    select * from test 格式:INSERT(str,pos,len,newstr) 解释: str:查询的例 pos:起始位置 len:从起始位置开始被后面newstr替换的长度 new ...

  7. windows 安装tensorflow

    原文知乎:https://zhuanlan.zhihu.com/p/25778703 前言 看到Rstudio中开始支持Tensorflow,本人是欣喜若狂的,同时TensorFlow官网从16年9月 ...

  8. QTP 自动化测试桌面程序--笔记(关闭 启动程序脚本) 、安装

    0 安装qtp .exe 文件 安装 插件文件(如delph) 1 关闭 启动程序: 将要操作的程序-存入localdatatable中 设置 迭代一次 rem SystemUtil.ClosePro ...

  9. Lodop文本项相对于文本框居中 两端对齐

    Lodop中ADD_PRINT_TEXT默认内容是相对于文本框居左的,如果想要设置相对于文本框居中,可用如下语句.还有一种是两端对齐,可以让内容的两端阿和文本框的最左和最右端对齐,文本项内容布满文本框 ...

  10. .net core 2.0 Autofac

    参考自 https://github.com/VictorTzeng/Zxw.Framework.NetCore 安装Autofac,在`project.csproj`加入 <PackageRe ...