bzoj3506 [Cqoi2014]排序机械臂
bzoj3506
此题是一道比较简单的spaly题目。
用splay维护序列,将每个点排到对应的位置之后删除,这样比较容易区间翻转。
我的指针写法在洛谷上AC了,但在bzoj上RE。
#include <cstdio>
#include <algorithm>
using namespace std;
inline int gi() {
register int a; register char c;
while(c=getchar(),c<'0');a=c-'0';
while(c=getchar(),'-'<c)a=(a<<3)+(a<<1)+c-'0';
return a;
}
#define N 100010
struct node {
int s; bool ly;
node*fa,*ch[2];
}MSET[N],*cnode=MSET,*mp[N],*nil,*rt;
pair<int,int>pr[N]; int rk[N];
inline void maintain(node*u) {u->s=1+u->ch[0]->s+u->ch[1]->s;}
inline void reverse(node*u) {u->ly^=1,swap(u->ch[0],u->ch[1]);}
inline void pushdown(node*u) {
if(!u->ly)return;
u->ly=false;
if(u->ch[0]!=nil)reverse(u->ch[0]);
if(u->ch[1]!=nil)reverse(u->ch[1]);
}
inline void rot(node*u) {
node*f=u->fa,*ff=f->fa;
int d=u==f->ch[1];
pushdown(f),pushdown(u);
if((f->ch[d]=u->ch[d^1])!=nil)f->ch[d]->fa=f;
if((u->fa=ff)!=nil)ff->ch[f==ff->ch[1]]=u;
u->ch[d^1]=f,f->fa=u;
maintain(f),maintain(u);
}
inline void splay(node*u,node*target) {
for(node*f;u->fa!=target;rot(u))
if((f=u->fa)->fa!=target)((u==f->ch[1])^(f==f->fa->ch[1]))?rot(u):rot(f);
if(target==nil) rt=u;
}
inline void del(node*u) {
splay(u,nil);
pushdown(u);
if(u->ch[0]==nil) rt=u->ch[1],rt->fa=nil;
else {
node*v=u->ch[0];
for(pushdown(v);v->ch[1]!=nil;pushdown(v=v->ch[1]));
splay(v,u);
if((v->ch[1]=u->ch[1])!=nil)v->ch[1]->fa=v;
v->fa=nil; maintain(rt=v);
}
}
node*build(int l,int r) {
if(l>r)return nil;
int m=l+r>>1;
node*u=cnode++;
mp[rk[m]]=u;
if((u->ch[0]=build(l,m-1))!=nil)u->ch[0]->fa=u;
if((u->ch[1]=build(m+1,r))!=nil)u->ch[1]->fa=u;
maintain(u);
}
int main() {
int n,i;
scanf("%d",&n);
for(i=1;i<=n;++i)scanf("%d",&pr[i].first),pr[i].second=i;
sort(pr+1,pr+n+1);
for(i=1;i<=n;++i)rk[pr[i].second]=i;
rk[n+1]=n+1;
nil=cnode++;
nil->ch[0]=nil->ch[1]=nil->fa=nil;
nil->s=nil->ly=0;
rt=build(0,n+1); rt->fa=nil;
for(i=1;i<=n;++i) {
splay(mp[i],nil);
printf("%d ",rt->ch[0]->s+i-1);
reverse(rt->ch[0]);
del(rt);
}return 0;
}
数组写法在bzoj也AC了。
#include <cstdio>
#include <algorithm>
using namespace std;
inline int gi() {
static int a; static char c;
while(c=getchar(),c<'0');a=c-'0';
while(c=getchar(),'-'<c)a=(a<<3)+(a<<1)+c-'0';
return a;
}
#define N 100010
int fa[N],ch[2][N],s[N],rt;
bool flag[N]; pair<int,int>a[N];
inline void reverse(const int&u) {flag[u]^=1; swap(ch[0][u],ch[1][u]);}
inline void down(const int&u) {
if(!flag[u])return;
flag[u]=false;
if(ch[0][u])reverse(ch[0][u]);
if(ch[1][u])reverse(ch[1][u]);
}
inline void up(const int&u) {s[u]=1+s[ch[0][u]]+s[ch[1][u]];}
inline void rot(const int&u) {
static int f,d,t;
f=fa[u],d=u==ch[1][f];
down(f),down(u);
(t=ch[d][f]=ch[d^1][u])?fa[t]=f:0;
(t=fa[u]=fa[f])?ch[f==ch[1][t]][t]=u:0;
fa[f]=u,ch[d^1][u]=f;
up(f),up(u);
}
inline void splay(int u,const int&target) {
for(int t;fa[u]^target;rot(u))
if(fa[t=fa[u]]^target) ((u==ch[1][t])^(t==ch[1][fa[t]]))?rot(u):rot(t);
if(!target) rt=u;
}
inline void del(int u) {
splay(u,0); down(u);
if(!ch[0][u]||!ch[1][u])fa[rt=ch[0][u]+ch[1][u]]=0;
else {
int t=ch[0][u];
for(down(t);ch[1][t];down(t))t=ch[1][t];
splay(t,u);
fa[ch[1][t]=ch[1][u]]=t;
fa[t]=0; rt=t;
up(t);
}
}
int build(int l,int r) {
if(l>r) return 0;
int m=l+r>>1;
s[m]=r-l+1;
if(ch[0][m]=build(l,m-1))fa[ch[0][m]]=m;
if(ch[1][m]=build(m+1,r))fa[ch[1][m]]=m;
return m;
}
int main() {
register int n=gi(),i;
for(i=1;i<=n;++i)a[i].first=gi(),a[i].second=i;
sort(a+1,a+n+1);
rt=build(1,n);
for(i=1;i<=n;++i) {
splay(a[i].second,0);
reverse(ch[0][rt]);
printf("%d ",s[ch[0][rt]]+i);
del(rt);
}
return 0;
}
bzoj3506 [Cqoi2014]排序机械臂的更多相关文章
- BZOJ1552[Cerc2007]robotic sort&BZOJ3506[Cqoi2014]排序机械臂——非旋转treap
题目描述 输入 输入共两行,第一行为一个整数N,N表示物品的个数,1<=N<=100000. 第二行为N个用空格隔开的正整数,表示N个物品最初排列的编号. 输出 输出共一行,N个用空格隔开 ...
- [BZOJ3506] [Cqoi2014] 排序机械臂 (splay)
Description 同OJ1552 Input Output Sample Input Sample Output HINT Source Solution Q:哎不是同一道题吗为什么分两篇博客来 ...
- [bzoj1552][Cerc2007]robotic sort&&[bzoj3506][Cqoi2014]排序机械臂
非常垃圾的一道平衡树,结果被日了一天.很难受嗷嗷嗷 首先不得不说网上的题解让我这个本来就不熟悉平衡树的彩笔很难受——并不好理解. 还好Sinogi大佬非常的神,一眼就切掉了,而且用更加美妙的解法. 题 ...
- 【BZOJ3506】[CQOI2014] 排序机械臂(Splay)
点此看题面 大致题意: 给你\(n\)个数.第一次找到最小值所在位置\(P_1\),翻转\([1,P_1]\),第二次找到剩余数中最小值所在位置\(P_2\),翻转\([2,P_2]\),以此类推.求 ...
- 【BZOJ3506】排序机械臂(Splay)
[BZOJ3506]排序机械臂(Splay) 题面 神TMBZOJ没有题面,感谢SYC的题面 洛谷的题面也不错 题解 对于每次旋转的物体 显然可以预处理出来 现在只要模拟旋转操作就行了 至于在哪里放标 ...
- P3165 [CQOI2014]排序机械臂
题目描述 为了把工厂中高低不等的物品按从低到高排好序,工程师发明了一种排序机械臂.它遵循一个简单的排序规则,第一次操作找到高度最低的物品的位置 P1P_1P1 ,并把左起第一个物品至 P1P_1P1 ...
- 洛谷P3165 [CQOI2014]排序机械臂
题目描述 为了把工厂中高低不等的物品按从低到高排好序,工程师发明了一种排序机械臂.它遵循一个简单的排序规则,第一次操作找到摄低的物品的位置P1,并把左起第一个至P1间的物品反序:第二次找到第二低的物品 ...
- BZOJ3506/1502 [CQOI2014]排序机械臂
传送门 依然是一道splay的区间操作,需要注意的是要把下标离散化后来表示splay的节点,我不知道怎么搞所以索性弄了个$ValuetoNode$,看样子没什么问题, 感觉他那个传下标的方法太暴力了. ...
- Luogu P3165 [CQOI2014]排序机械臂
先讲一下和这题一起四倍经验的题: Luogu P4402 [Cerc2007]robotic sort 机械排序 SP2059 CERC07S - Robotic Sort UVA1402 Robot ...
随机推荐
- Linux 学习
远程登录Linux(05) 文本方式远程: putty SecureCRT winSCP SshClient图形方式远程:Xmanager Xming ifconfigps -ef | gr ...
- DevExpress v16.1.5图表、Dashboard等多个控件API发生变化
Dashboard # BC3835: WinForms Dashboard Designer - ChartSeriesTypeGalleryItem bar item停用 在v16.1.5之前,D ...
- Appfuse:权限控制
Appfuse的权限控制依赖于Struts的Menu机制,common下的menu.jsp是对菜单顺序的定义,详细的菜单项和菜单链接及权限再menu-config.xml中控制,如下: <Men ...
- C#接口的显示和隐式实现
早上骑车上班走到半路发现手机忘带了,这年代兜里没装一分现金,吃饭都要刷手机,上班时间无聊了不能玩手机更是大问题,所以果断掉头拿手机.取完手机刚出门看这天阴沉沉的貌似要下雨,没雨衣,骑车又不能打伞,上次 ...
- http参数的封装(后台接受参数的场景)
场景 不管是任何web框架作为一个web的开发人员必须要搞明白control层如何接受各种参数. 下面就根据我们公司的系统架构(nutz)来进行一下场景描述.各位小伙伴也可以根据这些 场景自己去总结一 ...
- 机器学习实战笔记(Python实现)-08-线性回归
--------------------------------------------------------------------------------------- 本系列文章为<机器 ...
- JS原型链
JS作为发展了多年了对象语言,支持继承,和完全面向对象语言不同的是,JS依赖原型链来实现对象的继承. 首先JS的对象分两大类,函数对象和普通对象,每个对象均内置__proto__属性,在不人为赋值__ ...
- sipex3232上电发送输入端拉低问题
之前没怎么注意,在使用下载盒通过isp下载stm32时,由于是通过stm32板子给下载盒供电,所以是我的stm32板子和下载盒同时上电.这样总是不能够进入下载模式,后来对下载盒进行仿真,发现在下载发送 ...
- 昨天写支付接口时遇到支付接口返回数据接收地址,session数据丢失(或者说失效)的问题
在网上找了好久 才找到答案 分享给大家 http://www.zcool.com.cn/article/ZMzYwNTI=.html
- 在虚拟机中安装CentOS
1.准备工具 我当时下载的是VMware9.0.2,之后升级即可. 2.安装VMware9.0.2,按照步骤安装即可,安装成功并运行 选择创建新的虚拟机,出现下图,选择"自定义"后 ...