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 ...
随机推荐
- Android 命令行执行工具类
最近在做android项目的时候,需要执行命令行命令,之前在网上找的不仅杂乱而且错误多,于是自己写了一份. 话不多说,直接上代码 import android.util.Log; import jav ...
- Android中隐藏顶部状态栏的那些坑——Android开发之路3
Android中隐藏顶部状态栏的那些坑 先看看常规的隐藏状态栏的方法: 方法一: @Override protected void onCreate(Bundle savedInstanceState ...
- 【代码笔记】iOS-字符串的分割
一,代码. - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view. ...
- iOS7之后设置NavigationBar的背景
iOS7之后,请注意需要使用setBarTintColor ``` [self.navigationController.navigationBar setBarTintColor:[UIColor ...
- lucene自定义过滤器
先介绍下查询与过滤的区别和联系,其实查询(各种Query)和过滤(各种Filter)之间非常相似,可以这样说只要用Query能完成的事,用过滤也都可以完成,它们之间可以相互转换,最大的区别就是使用过滤 ...
- asp.net mvc 之旅 —— 第六站 ActionFilter的应用及源码分析
这篇文章我们开始看一下ActionFilter,从名字上其实就大概知道ActionFilter就是Action上的Filter,对吧,那么Action上的Filter大概有几个呢??? 这个问题其实还 ...
- Mysql查询语句使用select.. for update导致的数据库死锁分析
近期有一个业务需求,多台机器需要同时从Mysql一个表里查询数据并做后续业务逻辑,为了防止多台机器同时拿到一样的数据,每台机器需要在获取时锁住获取数据的数据段,保证多台机器不拿到相同的数据. 我们My ...
- Linux httpd源码编译安装
# wget http://apache.fayea.com/httpd/httpd-2.2.31.tar.bz2 去官网下载源码包 # mv httpd-.tar.bz2 /usr/local/sr ...
- ubuntu-Linux系统读取USB摄像头数据(gspca)
将摄像头图像保存为jpg格式.摄像头需要是gspca免驱的.uvc若用uvc格式的需要在图像中插入Huffman表.否则无法正常显示. 程序代码: #include <stdio.h> # ...
- 极路由2(极贰)在OpenWrt下定制自己的ss服务
默认刷入的OpenWrt带的ss, 只有ss-redir服务, 但是在实际使用中, 很多时候还是希望访问直接通过正常网关, 只有少部分访问需要通过ss, 所以希望能配置成为ss-local服务. 在保 ...