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] ...
随机推荐
- Nginx +tomcat 实现负载均衡集群
一. 工具 nginx-1.8.0 apache-tomcat-6.0.33 二. 目标 实现高性能负载均衡的Tomcat集群: 三. 步骤 1.首先下载Nginx ...
- react typescript 子组件调用父组件
//父组件 import * as React from 'react'import { Input } from 'antd'const Search = Input.Searchimport &q ...
- 18清明校内测试T3
扫雷(mine) Time Limit:1000ms Memory Limit:128MB 题目描述 rsy最近沉迷于一款叫扫雷的游戏. 这个游戏是这样的.一开始网格上有n*m个位置,其中有一些位 ...
- Luogu P2922 秘密消息
原题 P2922 [USACO08DEC]秘密消息Secret Message 题目描述 Bessie is leading the cows in an attempt to escape! To ...
- unigui菜单【3】
unigui菜单TuniTreeView 根据数据库表中的内容,显示菜单的处理: function TMainForm.CreateMenu: Integer; var myMenuPoint : P ...
- MongoDB整库备份与还原以及单个collection备份、恢复
备份前的检查> show dbsMyDB 0.0625GBadmin (empty)bruce 0.0625GBlocal (empty)test 0.0625GB> use MyDBsw ...
- toj 1421
题意:假如存在矩阵A,A[i][0] + A[i][1] + ...... + A[i][n - 1] == SR[i],A[0][j] + A[1][j] + ...... + A[n - 1][j ...
- nyoj_212_K尾相等数_210402272239
K尾相等数 时间限制:3000 ms | 内存限制:65535 KB 难度:1 描述 输入一个自然数K(K>1),如果存在自然数M和N(M>N),使得K^M和K^N均大于等于100 ...
- [luogu1373]小a和uim之大逃离_动态规划
小a和uim之大逃离 题目大意:有一个n*m的矩阵.每个格子上有一坨0~k不等量的权值.有两个人,每个人任选一个格子作为出发点,并只能向下或向右走.求最后两个人所得到的权值mod k相等的方案数. 注 ...
- Chrome development tools学习笔记(3)
(上次DOM的部分做了些补充,欢迎查看Chrome development tools学习笔记(2)) 利用DevTools Elements工具来调试页面样式 CSS(Cascading Style ...