BZOJ3506/1502 [CQOI2014]排序机械臂
依然是一道splay的区间操作,需要注意的是要把下标离散化后来表示splay的节点,我不知道怎么搞所以索性弄了个$ValuetoNode$,看样子没什么问题,
感觉他那个传下标的方法太暴力了..应该可以优化
//BZOJ 1552
//by Cydiater
//2016.9.7
#include <iostream>
#include <cstdio>
#include <cstring>
#include <string>
#include <algorithm>
#include <queue>
#include <map>
#include <ctime>
#include <cmath>
#include <iomanip>
#include <cstdlib>
using namespace std;
#define ll long long
#define up(i,j,n) for(int i=j;i<=n;i++)
#define down(i,j,n) for(int i=j;i>=n;i--)
;
const int oo=0x3f3f3f3f;
inline int read(){
,f=;
;ch=getchar();}
+ch-';ch=getchar();}
return x*f;
}
,root=,tol=,ValuetoNode[MAXN],q[MAXN],head,tail;
struct _data{
int id,v;
}a[MAXN];
struct SplayTree{
],fa,siz,v,tag;
}t[MAXN];
namespace solution{
inline ]==node;}
inline bool cmp(_data x,_data y){return x.v==y.v?x.id<y.id:x.v<y.v;}
inline bool re_cmp(_data x,_data y){return x.id<y.id;}
void updata(int node){
if(node){
t[node].siz=;
])t[node].siz+=t[t[node].son[]].siz;
])t[node].siz+=t[t[node].son[]].siz;
}
}
void downit(int node){
if(t[node].tag){
],rightson=t[node].son[];
;;
swap(t[node].son[],t[node].son[]);
t[node].tag=;
}
}
void rotate(int node){
int old=t[node].fa,oldf=t[old].fa,which=get(node);
t[old].son[which]=t[node].son[which^];t[t[old].son[which]].fa=old;
t[old].fa=node;t[node].son[which^]=old;t[node].fa=oldf;
]]=node;
updata(old);updata(node);
}
void build(int leftt,int rightt,int node,int fa){
if(leftt==rightt){
t[node].v=a[leftt].v;t[node].fa=fa;t[node].son[]=t[node].son[]=;
t[node].siz=;ValuetoNode[a[leftt].v]=node;t[node].tag=;
return;
}
t[node].tag=;
t[node].fa=fa;;
t[node].v=a[mid].v;ValuetoNode[a[mid].v]=node;
>=leftt){
t[node].son[]=++tol;build(leftt,mid-,tol,node);
}
<=rightt){
t[node].son[]=++tol;build(mid+,rightt,tol,node);
}
updata(node);
}
void splay(int node,int aim){
for(int fa;(fa=t[node].fa);rotate(node)){
if(node==aim)break;
if(t[node].fa==aim){
rotate(node);
break;
}
if(t[fa].fa==aim){
rotate(get(node)==get(fa)?fa:node);
rotate(node);
break;
}
if(t[fa].fa!=aim)rotate(get(node)==get(fa)?fa:node);
}
if(aim==root)root=node;
}
int find(int num){
int now=root;
){
downit(now);
]?t[t[now].son[]].siz:);
];
else{
)return now;
num-=(tmp+);
now=t[now].son[];
}
}
}
void init(){
N=read();
a[].v=oo;a[].id=;
up(i,,N+){
a[i].v=read();
a[i].id=i;
}N++;
a[++N].v=oo;a[N].id=N;
sort(a+,a+N+,cmp);
up(i,,N)a[i].v=++cnt;//sort by value
root=++tol;
sort(a+,a+N+,re_cmp);
build(,N,root,);
}
void slove(){
up(i,,N-){
int node=ValuetoNode[i];
head=;tail=;q[++tail]=node;;
int tmp=t[node].fa;
while(tmp!=root){
q[++tail]=tmp;
tmp=t[tmp].fa;
}q[++tail]=root;
while(head<=tail){
tmp=q[tail--];
downit(tmp);
}
splay(node,root);right_siz=t[t[root].son[]].siz+;
printf();)printf(" ");
int leftt=find(i),rightt=find(right_siz);
splay(leftt,root);splay(rightt,t[root].son[]);
];
t[Son].tag^=;
}
puts("");
}
}
int main(){
//freopen("input.in","r",stdin);
using namespace solution;
init();
slove();
;
}
BZOJ3506/1502 [CQOI2014]排序机械臂的更多相关文章
- 【BZOJ3506】[CQOI2014] 排序机械臂(Splay)
点此看题面 大致题意: 给你\(n\)个数.第一次找到最小值所在位置\(P_1\),翻转\([1,P_1]\),第二次找到剩余数中最小值所在位置\(P_2\),翻转\([2,P_2]\),以此类推.求 ...
- P3165 [CQOI2014]排序机械臂
题目描述 为了把工厂中高低不等的物品按从低到高排好序,工程师发明了一种排序机械臂.它遵循一个简单的排序规则,第一次操作找到高度最低的物品的位置 P1P_1P1 ,并把左起第一个物品至 P1P_1P1 ...
- 洛谷P3165 [CQOI2014]排序机械臂
题目描述 为了把工厂中高低不等的物品按从低到高排好序,工程师发明了一种排序机械臂.它遵循一个简单的排序规则,第一次操作找到摄低的物品的位置P1,并把左起第一个至P1间的物品反序:第二次找到第二低的物品 ...
- bzoj3506 [Cqoi2014]排序机械臂
bzoj3506 此题是一道比较简单的spaly题目. 用splay维护序列,将每个点排到对应的位置之后删除,这样比较容易区间翻转. 我的指针写法在洛谷上AC了,但在bzoj上RE. #include ...
- 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大佬非常的神,一眼就切掉了,而且用更加美妙的解法. 题 ...
- Luogu P3165 [CQOI2014]排序机械臂
先讲一下和这题一起四倍经验的题: Luogu P4402 [Cerc2007]robotic sort 机械排序 SP2059 CERC07S - Robotic Sort UVA1402 Robot ...
- 【洛谷 P3165】 [CQOI2014]排序机械臂 (Splay)
题目链接 debug了\(N\)天没debug出来,原来是找后继的时候没有pushdown... 众所周知,,Splay中每个编号对应的节点的值是永远不会变的,因为所有旋转.翻转操作改变的都是父节点和 ...
随机推荐
- 如何在Vue2中实现组件props双向绑定
Vue学习笔记-3 前言 Vue 2.x相比较Vue 1.x而言,升级变化除了实现了Virtual-Dom以外,给使用者最大不适就是移除的组件的props的双向绑定功能. 以往在Vue1.x中利用pr ...
- 从语言到库到框架,再到API,再到标记最后到DSL语言
计算机技术发展很快,而且越来越快,结果也是越来越复杂,那么我们到底怎么搞定复杂性并重用代码? 很明显,这是个大难题.一开始我们要解决计算问题,发展了基本的编程语言. 很快,编程语言不能满足需求,我们需 ...
- js中字符串和数组相互转化的方法
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px "Helvetica Neue"; color: #e4af0a } p. ...
- Hadoop2.6.0伪分布环境搭建
用到的软件: 一.安装jdk: 1.要安装的jdk,我把它拷在了共享文件夹里面. (用优盘拷也可以) 2.我把jdk拷在了用户文件夹下面. (其他地方也可以,不过路径要相应改变) 3.执行复制安装 ...
- Ubuntu更改右键菜单
方法/步骤1.这是我们在桌面文件夹ubuntugege上打开的右键菜单,你说你在~/.gnome2/nautilus-scripts/添加的右键菜单项目但它就是没有显示呀,于是你觉得Ubuntu 12 ...
- 关于#pragma once和#ifndefine组合的区别
最近在看duilib代码,发现头文件既有#pragma once 又有 #ifndefine...#define,忽然就觉得有点不解,因为据我所知这两者都是防止头文件二次包含的. 经过下面两位的解释后 ...
- [转]response.getWriter().write()与out.print()的区别
原文地址:http://blog.csdn.net/javaloveiphone/article/details/8133772 1.首先介绍write()和print()方法的区别: (1).w ...
- python中的字符数字之间的转换函数
int(x [,base ]) 将x转换为一个整数 long(x [,base ]) 将x转换为一个长整数 float(x ) ...
- poj1743 后缀数组求不可重叠的重复出现的子串最长长度
Musical Theme Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 25348 Accepted: 8546 De ...
- 【POJ 2503】Babelfish(字符串)
题 给定字典,再询问. 字典与询问之间有一个空行. cin.peek()是一个指针指向当前字符. #include<iostream> #include<string> #in ...