NOI2007项链工厂——sbTreap代码
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cstdlib> using namespace std;
struct node
{
int data;
int left;
int right;
int key;
int size;
bool turn;
int ans;
int mark;
node* ls,*rs;
node(int dat=,int ke=)
{
data=dat;
key=ke;
left=data;
right=data;
size=;
turn=false;
mark=;
ans=;
ls=NULL;
rs=NULL;
}
}no[]; void pushdown(node* now)
{
if (now->mark)
{
if (now->ls)
{
now->ls->data=now->mark;
now->ls->mark=now->mark;
now->ls->left=now->mark;
now->ls->right=now->mark;
now->ls->ans=;
}
if (now->rs)
{
now->rs->data=now->mark;
now->rs->mark=now->mark;
now->rs->left=now->mark;
now->rs->right=now->mark;
now->rs->ans=;
}
now->mark=;
}
if (now->turn)
{
node* l=now->ls;
node* r=now->rs;
if (now->ls)
now->ls->left^=now->ls->right^=now->ls->left^=now->ls->right;
if (now->rs)
now->rs->left^=now->rs->right^=now->rs->left^=now->rs->right;
now->ls=r;
now->rs=l;
if (now->ls) now->ls->turn=!now->ls->turn;
if (now->rs) now->rs->turn=!now->rs->turn;
now->turn=false;
}
} void update(node* now)
{
now->size=;
now->ans=;
now->left=now->data;
now->right=now->data;
if (now->ls)
{
now->size+=now->ls->size;
now->ans+=now->ls->ans;
now->left=now->ls->left;
}
if (now->rs)
{
now->size+=now->rs->size;
now->ans+=now->rs->ans;
now->right=now->rs->right;
}
if (now->ls)
if (now->ls->right==now->data) now->ans--;
if (now->rs)
if (now->rs->left==now->data) now->ans--;
} node* merge(node* a,node* b)
{
if (!b) return a;
if (!a) return b;
pushdown(a);
pushdown(b);
if (a->key<=b->key)
{
a->rs=merge(a->rs,b);
update(a);
return a;
}
else
{
b->ls=merge(a,b->ls);
update(b);
return b;
}
} struct npair
{
node* l,*r;
npair(node* a,node* b)
{
l=a;
r=b;
}
}; npair split(node* a,int k)
{
if (!a) return npair(NULL,NULL);
if (k==) return npair(NULL,a);
pushdown(a);
if (a->ls)
{
if (a->ls->size>=k)
{
npair km=split(a->ls,k);
a->ls=km.r;
update(a);
return npair(km.l,a);
}
else
{
npair km=split(a->rs,k-a->ls->size-);
a->rs=km.l;
update(a);
return npair(a,km.r);
}
}
else
{
npair km=split(a->rs,k-);
a->rs=km.l;
update(a);
return npair(a,km.r);
}
}
node* insert(node* root,node* newnode)
{
return merge(root,newnode);
} void turn(node* now)
{
now->turn=!now->turn;
now->left^=now->right^=now->left^=now->right;
} node* rotate(node* root,int num)
{
int n=root->size;
num=num%n;
int k=n-num;
npair km = split(root,k);
return merge(km.r,km.l);
} node* flip(node* root)
{
int n=root->size;
int r;
if (n%==)
{
r=n/+;
}
else
{
r=n/;
}
npair km=split(root,r);
npair km2=split(km.l,);
if (n%==)
{
turn(km2.r);
turn(km.r);
return merge(merge(km2.l,km.r),km2.r);
}
else
{
npair km3=split(km.r,);
turn(km2.r);
turn(km3.r);
return merge(merge(km2.l,km3.r),merge(km3.l,km2.r));
}
} node* swap(node* root,int i,int j)
{
if (i>j) i^=j^=i^=j;
if (i==j) return root;
npair km=split(root,i);
npair km2=split(km.l,i-);
npair km3=split(km.r,j-i);
npair km4=split(km3.l,j-i-);
return merge(merge(merge(km2.l,km4.r),km4.l),merge(km2.r,km3.r));
} node* paint(node* root,int i,int j,int x)
{
int n=root->size;
if (i<=j)
{
npair km=split(root,i-);
npair km2=split(km.r,j-i+);
km2.l->mark=x;
km2.l->data=x;
km2.l->ans=;
km2.l->left=x;
km2.l->right=x;
return merge(km.l,merge(km2.l,km2.r));
}
else
{
npair km=split(root,j);
int nn=km.r->size;
npair km2=split(km.r,nn-n+i-);
km.l->mark=x;
km.l->data=x;
km.l->ans=;
km.l->left=x;
km.l->right=x;
km2.r->mark=x;
km2.r->data=x;
km2.r->ans=;
km2.r->left=x;
km2.r->right=x;
return merge(km.l,merge(km2.l,km2.r));
}
} node* root;
int countS(int i,int j)
{
int n=root->size;
if (i<=j)
{
npair km=split(root,i-);
npair km2=split(km.r,j-i+);
int ret=km2.l->ans;
root=merge(km.l,merge(km2.l,km2.r));
return ret;
}
else
{
npair km=split(root,j);
int nn=km.r->size;
npair km2=split(km.r,nn-n+i-);
int ret=km.l->ans+km2.r->ans;
if (km.l->left==km2.r->right) ret--;
root=merge(km.l,merge(km2.l,km2.r));
return ret;
}
} int count()
{
int ret=root->ans;
if (root->left==root->right&&ret!=) ret--;
return ret;
} void print(node* now,bool b)
{
if (!now) return;
b=b^now->turn;
// if (!b)
print(now->ls,b);
// else
// print(now->rs,b);
printf("data: %d size: %d mark: %d turn: %d ans: %d left: %d right: %d\n",now->data,now->size,now->mark,now->turn,now->ans,now->left,now->right);
// if (!b)
print(now->rs,b);
// else
// print(now->ls,b);
} void print(node* now)
{
if (!now) return;
pushdown(now);
print(now->ls);
printf("%d\n",now->data);
print(now->rs);
} int cnt=-;
int main()
{
int n,c;
scanf("%d%d",&n,&c);
int j;
for (int i=;i<=n;++i)
{
scanf("%d",&j);
node* q=&no[++cnt];
*q=node(j,rand());
root=insert(root,q);
}
int m;
scanf("%d",&m);
char cmd[];
int l,r,k;
for (int i=;i<=m;++i)
{
scanf("%s",cmd);
if (cmd[]=='R'){scanf("%d",&k);root=rotate(root,k);}
if (cmd[]=='F'){root=flip(root);}
if (cmd[]=='S'){scanf("%d%d",&l,&r);root=swap(root,l,r);}
if (cmd[]=='P'){scanf("%d%d%d",&l,&r,&k);root=paint(root,l,r,k);}
if (cmd[]=='C'&&cmd[]!='S'){printf("%d\n",count());}
if (cmd[]=='C'&&cmd[]=='S'){scanf("%d%d",&l,&r);printf("%d\n",countS(l,r));}
// printf("%d\n",i);
// printf("--------------------\n");
// print(root,false);
// printf("--------------------\n");
}
return ;
}
NOI2007项链工厂——sbTreap代码的更多相关文章
- bzoj 1493: [NOI2007]项链工厂(线段树)
1493: [NOI2007]项链工厂 Time Limit: 30 Sec Memory Limit: 64 MBSubmit: 1256 Solved: 545[Submit][Status] ...
- BZOJ_1493_[NOI2007]项链工厂_Splay
BZOJ_1493_[NOI2007]项链工厂_Splay Description T公司是一家专门生产彩色珠子项链的公司,其生产的项链设计新颖.款式多样.价格适中,广受青年人的喜爱. 最近T公司打算 ...
- BZOJ1493 [NOI2007]项链工厂
未完待续... 终于改对了 热泪盈眶.jpg 错误原因:pushdown的时候没有判断是否有左右儿子,也没当x=0 return,于是出现一些奇怪的错误 #include<bits/stdc++ ...
- 数据结构(Splay平衡树): [NOI2007] 项链工厂
[NOI2007] 项链工厂 ★★★ 输入文件:necklace.in 输出文件:necklace.out 简单对比 时间限制:4 s 内存限制:512 MB [问题描述] T公司是一 ...
- bzoj1493[NOI2007]项链工厂 线段树
1493: [NOI2007]项链工厂 Time Limit: 30 Sec Memory Limit: 64 MBSubmit: 1712 Solved: 723[Submit][Status] ...
- BZOJ1493 NOI2007 项链工厂 线段树模拟
提交地址:http://www.lydsy.com/JudgeOnline/problem.php?id=1493 题目大意:给一个数列,进行一系列操作.包括旋转,翻转,改变等操作,以及查询颜色段数. ...
- 1493: [NOI2007]项链工厂
线段树. 真还就是个线段树.. 除去操作1,2的话,线段树很容易就处理了,问题在于如何处理操作1和2.(这点没想到).. 我们用一个delta维护操作1,如果没有旋转就+k,不然就-k. 每次读入i和 ...
- NOI2007 项链工厂
题目链接:戳我 60pts 有一点容易写错的小细节: 比如说求全局的段数的时候,如果只有一种颜色,那么当左右端点相等时,就不要ans--了. 注意右端点小于左端点的情况. #include<io ...
- 【BZOJ-1493】项链工厂 Splay
1493: [NOI2007]项链工厂 Time Limit: 30 Sec Memory Limit: 64 MBSubmit: 1440 Solved: 626[Submit][Status] ...
随机推荐
- 谷歌浏览器中a:link设置字体颜色无效问题
<div id="box"> <a href="#">111111</a> <a href=""& ...
- CSS 选择器 知识点
<html> <head> <style type="text/css"> h1 > strong { /*子元素选择器 只选择自己 的子 ...
- [LUOGU] 4933 大师
\(Orz\) \(ljt12138!\) 设状态\(f[i][j]\)表示以\(i\)为结尾,公差为\(j\)的长度大于\(1\)的数列有几个. 然后转移方程就很好想了. \(k=H[i]-H[j] ...
- jdk编译安装及tomcat编译安装
这里我安装的jdk版本为1.8版本,tomcat版本为8.5(请上官网下载) 运维开发技术交流群欢迎大家加入一起学习(QQ:722381733) jdk部署: 1.前往软件所在路径 [root@web ...
- Linux 实用指令(4)
目录 实用指令 1.指定运行级别 2.切换到指定运行级别的指令 3.帮助指令 3.1man获得帮助信息 3.2help指令 4.文件目录类 4.1pwd指令 4.2 ls指令 4.3 cd指令 4.4 ...
- 利用Linux中的crontab实现分布式项目定时任务
@Controller @RequestMapping("/task/topic") public class TopicQuartzController { protected ...
- [POJ1226]Substrings(后缀数组)
传送门 给定 n 个字符串,求出现或反转后出现在每个字符串中的最长子串. 算法分析: 这题不同的地方在于要判断是否在反转后的字符串中出现.其实这并没有加大题目的难度. 只需要先将每个字符串都反过来写一 ...
- HDU 5433
每次BC都好心酸... BFS+queue..状态可以设为p_val[x][y][k],加上斗志的值. #include <iostream> #include <cstdio> ...
- gradle配置国内的镜像
gradle配置国内的镜像 学习了:http://blog.csdn.net/stdupanda/article/details/72724181 http://blog.csdn.net/lj402 ...
- Swift和Objective-C混合编程——Swift调用OC
在iOS应用的开发中.Swift必将代替OC,两者的趋势是"短期共存,长期代替".但曾经有太多的代码是用OC语言完毕的,而Swift的开发也从 OC中继承了非常多的特性.两者也是有 ...