3223: Tyvj 1729 文艺平衡树

Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 6881  Solved: 4213
[Submit][Status][Discuss]

Description

 
您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1

Input

第一行为n,m n表示初始序列有n个数,这个序列依次是(1,2……n-1,n)  m表示翻转操作次数
接下来m行每行两个数[l,r] 数据保证 1<=l<=r<=n

Output

输出一行n个数字,表示原始序列经过m次变换后的结果

Sample Input

5 3

1 3

1 3

1 4

Sample Output

4 3 2 1 5

HINT

N,M<=100000

模板题,区间翻转问题

延时标记的作用是优化,如果一个区间翻转之后再翻转回来,用延时标记就可以优化,不必再翻转。比如翻转[1,4],再翻转[1,4],就可以延时标记优化。

其他的代码里写了注释。

代码:

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<bitset>
#include<cassert>
#include<cctype>
#include<cmath>
#include<cstdlib>
#include<ctime>
#include<deque>
#include<iomanip>
#include<list>
#include<map>
#include<queue>
#include<set>
#include<stack>
#include<vector>
using namespace std;
typedef long long ll; const double PI=acos(-1.0);
const double eps=1e-;
const ll mod=1e9+;
const int inf=0x3f3f3f3f;
const int maxn=1e5+;
const int maxm=+;
#define ios ios::sync_with_stdio(false);cin.tie(0);cout.tie(0); /*
将当前排名为l-1 +1 的节点转到根
将当前排名为r+2的节点转到根的右子树的根节点
则根的右子树的根节点的左子树为所求区间
直接打标记就可以了
*/ int n,m,sz,rt,pre[maxn],l,r,ch[maxn][],data[maxn],size[maxn],rev[maxn]; //在爸爸节点打上标记,然后进行下放,如果进行了两次相反的翻转,lazy标记就会消失,这样就减少了翻转次数达到优化 void pushup(int k)//要先pushup儿子才能pushup爸爸
{
size[k]=size[ch[k][]]+size[ch[k][]]+;//当前节点的size为左子树+右子树+自己
} void pushdown(int k)//要先pushdown爸爸才能pushdown儿子
{
int l=ch[k][],r=ch[k][];//左儿子和右儿子
if(rev[k]){//翻转区间
swap(ch[k][],ch[k][]);//翻转左右儿子
rev[l]^=;rev[r]^=;//标记下传
rev[k]=;//当前节点标记去掉
}
} void rotate(int x,int &k)//翻转操作
{
int y=pre[x],z=pre[y],l,r;
if(ch[y][]==x) l=;
else l=;
r=l^;
if(y==k) k=x;
else{if(ch[z][]==y) ch[z][]=x;else ch[z][]=x;}
pre[x]=z;pre[y]=x;pre[ch[x][r]]=y;
ch[y][l]=ch[x][r];ch[x][r]=y;
pushup(y);pushup(x);
} void splay(int x,int &k)//splay到目标状态
{
while(x!=k){
int y=pre[x],z=pre[y];
if(y!=k){
if(ch[y][]==x^ch[z][]==y)rotate(x,k);
else rotate(y,k);
}
rotate(x,k);
}
} int find(int k,int rank)
{
pushdown(k);//有标记就pushdown
int l=ch[k][],r=ch[k][];
if(size[l]+==rank) return k;
else if(size[l]>=rank) return find(l,rank);
else return find(r,rank-size[l]-);
} void change(int l,int r)
{
int x=find(rt,l),y=find(rt,r+);
splay(x,rt);splay(y,ch[x][]);
int z=ch[y][];
rev[z]^=;
} void build(int l,int r,int f)
{
if(l>r) return;
int now=data[l],last=data[f];
if(l==r){
pre[now]=last;
size[now]=;
if(l<f) ch[last][]=now;
else ch[last][]=now;
return;
} int mid=(l+r)>>;
now=data[mid];
build(l,mid-,mid);
build(mid+,r,mid);
pre[now]=last;
pushup(mid);
if(mid<f) ch[last][]=now;
else ch[last][]=now;
} int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=n+;i++)
data[i]=++sz;
build(,n+,);//建树,建两个哨兵节点为1,n+2。
rt=(n+)>>;//中点为rt
for(int i=;i<=m;i++){
scanf("%d%d",&l,&r);
change(l,r);
}
for(int i=;i<=n+;i++)
printf("%d ",find(rt,i)-);//去掉哨兵节点
return ;
}

先贴个板子,有的操作并不理解,过几天再看。

BZOJ 3223: Tyvj 1729 文艺平衡树-Splay树(区间翻转)模板题的更多相关文章

  1. [BZOJ 3223 & Tyvj 1729]文艺平衡树 & [CodeVS 3243]区间翻转

    题目不说了,就是区间翻转 传送门:BZOJ 3223 和 CodeVS 3243 第一道题中是1~n的区间翻转,而第二道题对于每个1~n还有一个附加值 实际上两道题的思路是一样的,第二题把值对应到位置 ...

  2. BZOJ 3223: Tyvj 1729 文艺平衡树(splay)

    速度居然进前十了...第八... splay, 区间翻转,用一个类似线段树的lazy标记表示是否翻转 ------------------------------------------------- ...

  3. bzoj 3223: Tyvj 1729 文艺平衡树 (splay)

    链接: https://www.lydsy.com/JudgeOnline/problem.php?id=3223 题面: 3223: Tyvj 1729 文艺平衡树 Time Limit: 10 S ...

  4. bzoj 3223/tyvj 1729 文艺平衡树 splay tree

    原题链接:http://www.tyvj.cn/p/1729 这道题以前用c语言写的splay tree水过了.. 现在接触了c++重写一遍... 只涉及区间翻转,由于没有删除操作故不带垃圾回收,具体 ...

  5. BZOJ - 3223 Tyvj 1729 文艺平衡树 (splay/无旋treap)

    题目链接 splay: #include<bits/stdc++.h> using namespace std; typedef long long ll; ,inf=0x3f3f3f3f ...

  6. BZOJ 3223 Tyvj 1729 文艺平衡树 | Splay 维护序列关系

    题解: 每次reverse(l,r) 把l-1转到根,r+1变成他的右儿子,给r+1的左儿子打个标记就是一次反转操作了 每次find和dfs输出的时候下放标记,把左儿子和右儿子换一下 记得建树的时候建 ...

  7. BZOJ 3223: Tyvj 1729 文艺平衡树

    3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3628  Solved: 2052[Submit][Sta ...

  8. BZOJ 3223 Tyvj 1729 文艺平衡树(Splay)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3223 [题目大意] 给出一数列,问m次区间翻转后的结果. [题解] Splay 区间翻 ...

  9. fhq_treap || BZOJ 3223: Tyvj 1729 文艺平衡树 || Luogu P3391 【模板】文艺平衡树(Splay)

    题面: [模板]文艺平衡树(Splay) 题解:无 代码: #include<cstdio> #include<cstring> #include<iostream> ...

随机推荐

  1. Session接口常用方法

    org.hibernate.Session接口 beginTransaction 开启事务 clear 清缓存 close 关闭session connection - 过时 获取Connection ...

  2. XJOI NOIP模拟题2

    第一题 组合计数 分析: 从前往后一位一位的计算 先算第一位比t小的数目,再算第一位与t[1]相同,第2位比t小的个数以此类推 先预处理一个数组h,h[i]表示从1~it串与s串不同的位数 对于第i位 ...

  3. BZOJ4424/CF19E Fairy(dfs树+树上差分)

    即删除一条边使图中不存在奇环.如果本身就是个二分图当然任意一条边都可以,先check一下.否则肯定要删除在所有奇环的交上的边. 考虑怎么找这些边.跑一遍dfs造出dfs树,找出返祖边构成的奇环.可以通 ...

  4. HDU3338:Kakuro Extension(最大流)

    Kakuro Extension Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  5. 1040: [ZJOI2008]骑士~基环外向树dp

    Z国的骑士团是一个很有势力的组织,帮会中汇聚了来自各地的精英.他们劫富济贫,惩恶扬善,受到社会各界的赞扬.最近发生了一件可怕的事情,邪恶的Y国发动了一场针对Z国的侵略战争.战火绵延五百里,在和平环境中 ...

  6. codeforces902B. Coloring a Tree

    B. Coloring a Tree 题目链接: https://codeforces.com/contest/902/problem/B 题意:给你一颗树,原先是没有颜色的,需要你给树填色成指定的样 ...

  7. 最小k度最小生成树模板

    代码是抄的 题解是瞄的 可我想学习的心是真的嘤嘤嘤 然而 还是上传一份ioi大神的论文吧 链接:https://pan.baidu.com/s/1neIW9QeZEa0hXsUqJTjmeQ 密码:b ...

  8. VC++使用CImage PNG转BMP图片透明背景处理

    PNG格式的图片是支持透明通道的,BMP格式的图片是没有透明通道的,所以当PNG格式的图片转换为BMP格式时,对于PNG图片的透明背景就需要进行特别的处理. VC++中的HBITMAP是支持透明色的, ...

  9. 获取html元素内容

    html: <!DOCTYPE ><html> <head> <meta http-equiv="Content-Type" conten ...

  10. js和jquery修改背景颜色的区别

    html: <HTML> <head> <meta http-equiv="content-type" content="text/html ...