HNOI2017单旋
单旋
- 这道题做法贼多,LCT,splay,线段树什么的貌似都行。
- 像我这种渣渣只会线段树了(高级数据结构学了也不会用)。
- 首先离线所有操作,因为不会有两个点值重复,所以直接离散。
- 一颗线段树来维护所有点的深度,并将所有值丢进\(set\)中。
- 插入操作,在set找到前驱后继,前驱没有右儿子就放前驱右儿子,否则放后继左儿子,同时用\(ch\)和\(fa\)假装模拟树的形态。
- 旋转操作,在\(set\)里找到节点,可以发现旋转操作该点儿子深度不变,其他点深度加一,处理一下父子关系,然后线段树修改区间即可。
- 删除操作,先旋转操作,\(root\)一定只有一个儿子,直接删掉\(root\)即可,将所有点深度减一
// luogu-judger-enable-o2
#include<bits/stdc++.h>
using namespace std;
typedef int sign;
typedef long long ll;
#define For(i,a,b) for(register sign i=(sign)a;i<=(sign)b;++i)
#define Fordown(i,a,b) for(register sign i=(sign)a;i>=(sign)b;--i)
const int N=1e5+5;
bool cmax(sign &a,sign b){return (a<b)?a=b,1:0;}
bool cmin(sign &a,sign b){return (a>b)?a=b,1:0;}
template<typename T>inline T read()
{
T f=1,ans=0;
char ch=getchar();
while(!isdigit(ch)&&ch!='-')ch=getchar();
if(ch=='-')f=-1,ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch-'0'),ch=getchar();
return ans*f;
}
template<typename T>inline void write(T x,char y)
{
if(x==0)
{
putchar('0');putchar(y);
return;
}
if(x<0)
{
putchar('-');
x=-x;
}
static char wr[20];
int top=0;
for(;x;x/=10)wr[++top]=x%10+'0';
while(top)putchar(wr[top--]);
putchar(y);
}
void file()
{
#ifndef ONLINE_JUDGE
freopen("3721.in","r",stdin);
freopen("3721.out","w",stdout);
#endif
}
int n,opt[N][2];
int a[N],top;
void input()
{
n=read<int>();
For(i,1,n)
{
opt[i][0]=read<int>();
if(opt[i][0]==1)a[++top]=read<int>(),opt[i][1]=a[top];
}
}
namespace Tree
{
#define mid ((l+r)>>1)
#define lson h<<1,l,mid
#define rson h<<1|1,mid+1,r
ll lazy[N<<2],sum[N<<2];
void push_down(int h,int l,int r)
{
if(!lazy[h])return;
int ls=h<<1,rs=ls|1;
lazy[ls]+=lazy[h];lazy[rs]+=lazy[h];
sum[ls]+=lazy[h]*1ll*(mid-l+1);
sum[rs]+=lazy[h]*1ll*(r-mid);
lazy[h]=0;
}
void push_up(int h)
{
sum[h]=sum[h<<1]+sum[h<<1|1];
}
void update(int h,int l,int r,int s,int t,int v)
{
if(s<=l&&r<=t)
{
lazy[h]+=v;
sum[h]+=1ll*v*1ll*(r-l+1);
}
else
{
push_down(h,l,r);
if(s<=mid)update(lson,s,t,v);
if(mid<t)update(rson,s,t,v);
push_up(h);
}
}
int query(int h,int l,int r,int pos)
{
if(l==r)return sum[h];
push_down(h,l,r);
int res;
if(pos<=mid)res=query(lson,pos);
else res=query(rson,pos);
push_up(h);
return res;
}
void modify(int h,int l,int r,int pos,int v)
{
if(l==r)sum[h]=v;
else
{
push_down(h,l,r);
if(pos<=mid)modify(lson,pos,v);
else modify(rson,pos,v);
push_up(h);
}
}
}
set<int>s;
set<int>::iterator it;
int root,ch[N][2],fa[N];
typedef pair<int,int>pii;
#define fir first
#define sec second
void insert(int x)
{
int pre,predep;
it=s.insert(x).fir;
if(!root)
{
root=x;
Tree::modify(1,1,n,x,1);
puts("1");
return;
}
if(it==s.begin())
{
++it;pre=*it;
fa[x]=pre;ch[pre][0]=x;
}
else
{
--it;
if(!ch[*it][1])
{
pre=*it;
fa[x]=pre;
ch[pre][1]=x;
}
else
{
++it;++it;
pre=*it;
fa[x]=pre;
ch[pre][0]=x;
}
}
predep=Tree::query(1,1,n,pre);
write(predep+1,'\n');
Tree::modify(1,1,n,x,predep+1);
}
void clear(int x)
{
ch[x][0]=ch[x][1]=fa[x]=0;
}
void rotate_min()
{
it=s.begin();
int x=*it;
if(x==root){puts("1");return;}
int y=ch[x][1],oldroot=root,pre=fa[x];
write(Tree::query(1,1,n,x),'\n');
if(y)Tree::update(1,1,n,x+1,fa[x]-1,-1);
Tree::update(1,1,n,1,n,1);
Tree::modify(1,1,n,x,1);
root=x;
ch[x][1]=oldroot;fa[oldroot]=x;
ch[pre][0]=y;fa[y]=pre;
}
void del_min()
{
rotate_min();
s.erase(root);
Tree::update(1,1,n,1,n,-1);
if(!ch[root][0]&&!ch[root][1])clear(root),root=0;
else
{
int oldroot=root;
root=ch[root][1];
clear(oldroot);fa[root]=0;
}
}
void rotate_max()
{
it=s.end();--it;
int x=*it;
if(x==root){puts("1");return;}
write(Tree::query(1,1,n,x),'\n');
int y=ch[x][0],oldroot=root,pre=fa[x];
if(y)Tree::update(1,1,n,fa[x]+1,x-1,-1);
Tree::update(1,1,n,1,n,1);
Tree::modify(1,1,n,x,1);
root=x;
fa[oldroot]=x;ch[x][0]=oldroot;
ch[pre][1]=y;fa[y]=pre;
}
void del_max()
{
rotate_max();
s.erase(root);
Tree::update(1,1,n,1,n,-1);
if(!ch[root][0]&&!ch[root][1])clear(root),root=0;
else
{
int oldroot=root;
root=ch[root][0];
clear(oldroot);fa[root]=0;
}
}
void work()
{
sort(a+1,a+top+1);
For(i,1,n)
{
if(opt[i][0]==1)
{
opt[i][1]=lower_bound(a+1,a+top+1,opt[i][1])-a;
insert(opt[i][1]);
}
else if(opt[i][0]==2)rotate_min();
else if(opt[i][0]==3)rotate_max();
else if(opt[i][0]==4)del_min();
else if(opt[i][0]==5)del_max();
}
}
int main()
{
file();
input();
work();
return 0;
}
HNOI2017单旋的更多相关文章
- bzoj 4825: [Hnoi2017]单旋 [lct]
4825: [Hnoi2017]单旋 题意:有趣的spaly hnoi2017刚出来我就去做,当时这题作死用了ett,调了5节课没做出来然后发现好像直接用lct就行了然后弃掉了... md用lct不知 ...
- 【LG3721】[HNOI2017]单旋
[LG3721][HNOI2017]单旋 题面 洛谷 题解 20pts 直接模拟\(spaly\)的过程即可. 100pts 可以发现单旋最大.最小值到根,手玩是有显然规律的,发现只需要几次\(lin ...
- 4825: [Hnoi2017]单旋
4825: [Hnoi2017]单旋 链接 分析: 以后采取更保险的方式写代码!!!81行本来以为不特判也可以,然后就总是比答案大1,甚至出现负数,调啊调啊调啊调~~~ 只会旋转最大值和最小值,以最小 ...
- [BZOJ4825][HNOI2017]单旋(线段树+Splay)
4825: [Hnoi2017]单旋 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 667 Solved: 342[Submit][Status][ ...
- 【BZOJ4825】[Hnoi2017]单旋 线段树+set
[BZOJ4825][Hnoi2017]单旋 Description H 国是一个热爱写代码的国家,那里的人们很小去学校学习写各种各样的数据结构.伸展树(splay)是一种数据结构,因为代码好写,功能 ...
- bzoj4825 [Hnoi2017]单旋
Description H 国是一个热爱写代码的国家,那里的人们很小去学校学习写各种各样的数据结构.伸展树(splay)是一种数据结构,因为代码好写,功能多,效率高,掌握这种数据结构成为了 H 国的必 ...
- BZOJ:4825: [Hnoi2017]单旋
Description H 国是一个热爱写代码的国家,那里的人们很小去学校学习写各种各样的数据结构.伸展树(splay)是一种数据结构,因为代码好写,功能多,效率高,掌握这种数据结构成为了 H 国的必 ...
- HNOI2017 单旋
题目描述 网址:https://www.luogu.org/problemnew/show/3721 大意: 有一颗单旋Splay(Spaly),以key值为优先度,总共有5个操作. [1] 插入一个 ...
- P3721 [AH2017/HNOI2017]单旋
题目:https://www.luogu.org/problemnew/show/P3721 手玩一下即可AC此题. 结论:插入x后,x要么会成为x的前驱的右儿子,要么成为x的后继的左儿子,这取决于它 ...
随机推荐
- sql储存过程in(多个参数)
一.用sql函数 首先要创建一个截取字符串的函数,新建一个查询,把下面代码复制进去执行. 函数SqlitIn的第一个参数是储存过程要in的字符串,第二个参数是分隔符 CREATE function S ...
- [C#]使用Windows Form开发的百度网盘搜索工具
BaiduDiskSearcher 用C#编写的百度网盘搜索工具(.NET4.0 & Visual Studio 2017) 功能 1.搜索任意名称的百度网盘共享文件 2.可以左键双击打开网盘 ...
- Spring学习日志之纯Java配置的MVC框架搭建
依赖引入 <dependencies> <dependency> <groupId>javax.servlet</groupId> <artifa ...
- 一个很好用的在线编辑、展示、分享、交流JavaScript 代码的平台
在发表博客时,有一些代码只能粘贴进去,而不能看到代码运行的效果,需要读者把代码粘贴进自己的编辑器,然后再运行看效果,这是一件很耗时的事情 在平时百度的时候,我发现一些网站可以在线预览功能,而且可以在线 ...
- Gerrit上分支操作记录(创建分支、删除分支)
Git分支对于一个项目的代码管理而言,是十分重要的!许多久用git的朋友可能已经掌握的很牢固了,但对于一些初涉git的童鞋来说,可能还不是很熟悉.在此,我将自己的一些操作经历做一梳理,希望能帮助到有用 ...
- 嵌入式linux教程
串口通信minicom $ sudo apt-get install minicom ///安装 # minicom –s //运行 //CTRL+A Z 弹出菜单 2.NFS网络文件配置 ...
- HDOJ1287_破译密码
一道正常简单题 曲折解题 做这题的时候看了很久没有看懂是什么意思,最后以为是一道单独的数学题把B这个大写字母猜出来进行异或运算,还不知道C里面异或运算可以直接有符号的:),导致又去学习了一下十进制转换 ...
- Maximal Binary Matrix CodeForces - 803A (贪心+实现)
题目链接 题意有点坑: 给你一个N*N的矩阵,让你填入K个1,使之整个矩阵关于左上到右下的对角线对称,并且这个要求这个矩阵的字典序最大. 对矩阵的字典序的定义是从每一行的第一个元素开始比较,大着为字典 ...
- Javascript中的闭包和C#中的闭包
js和C#中都有闭包的概念,闭包本质上是一个对象,是指有权访问另一个局部作用域中的变量的对象(或函数,在C#中是委托).这个对象和函数/方法有关: 在js中,闭包是由于函数引用了局部变量形成的.在C# ...
- mysql复杂查询1
https://blog.csdn.net/fly910905/article/details/79846949