题意

所谓线段树分裂其实是本题的在线做法。

考虑如果我们有一个已经排好序的区间的权值线段树,那么就可以通过线段树上二分的方法得到第\(k\)个数是谁。

于是用set维护每个升序/降序区间的左右端点以及对应的线段树根节点,区间排序就将区间拆出来,并将对应的线段树也拆出来。

拆线段树就是将前k个值建一棵新树拆出来,用类似fhq treap的方法即可。

code:

#include<bits/stdc++.h>
using namespace std;
#define lc(p) (seg[p].lc)
#define rc(p) (seg[p].rc)
#define sum(p) (seg[p].sum)
const int maxn=1e5+10;
int n,m,Q,tot;
int a[maxn];
queue<int>pool;
struct Seg{int lc,rc,sum;}seg[maxn*20];
struct node
{
int l,r,root,op;
bool operator<(const node& x)const{return r==x.r?l<x.l:r<x.r;}
};
set<node>s;
inline int read()
{
char c=getchar();int res=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9')res=res*10+c-'0',c=getchar();
return res*f;
}
inline int New()
{
int x;
if(!pool.empty())x=pool.front(),pool.pop();
else x=++tot;
lc(x)=rc(x)=sum(x)=0;
return x;
}
void insert(int &p,int l,int r,int pos)
{
if(!p)p=New();
sum(p)++;
if(l==r)return;
int mid=(l+r)>>1;
if(pos<=mid)insert(lc(p),l,mid,pos);
else insert(rc(p),mid+1,r,pos);
}
void split(int &p,int q,int l,int r,int k)
{
if(!k)return;
if(!p)p=New();
sum(p)+=k,sum(q)-=k;
if(l==r)return;
int mid=(l+r)>>1,tmp=sum(lc(q));
if(k<tmp)split(lc(p),lc(q),l,mid,k);
else
{
lc(p)=lc(q),lc(q)=0;
split(rc(p),rc(q),mid+1,r,k-tmp);
}
}
int merge(int p,int q)
{
if(!p||!q)return p+q;
lc(p)=merge(lc(p),lc(q));rc(p)=merge(rc(p),rc(q));
sum(p)+=sum(q);
pool.push(q);
return p;
}
int find(int p,int l,int r,int k)
{
if(l==r)return l;
int mid=(l+r)>>1;
if(sum(lc(p))>=k)return find(lc(p),l,mid,k);
else return find(rc(p),mid+1,r,k-sum(lc(p)));
}
inline int nodesplit(int l,int r)
{
set<node>::iterator it=s.lower_bound((node){0,l,0,0});
if((it->l)!=l)
{
node now=*it;
int p=0;
s.erase(it);
if(!now.op)
{
split(p,now.root,1,n,l-now.l);
s.insert((node){now.l,l-1,p,0});
s.insert((node){l,now.r,now.root,0}); }
else
{
split(p,now.root,1,n,now.r-l+1);
s.insert((node){now.l,l-1,now.root,1});
s.insert((node){l,now.r,p,1});
}
}
it=s.lower_bound((node){0,r,0,0});
if((it->r)!=r)
{
node now=*it;
int p=0;
s.erase(it);
if(!now.op)
{
split(p,now.root,1,n,r-now.l+1);
s.insert((node){now.l,r,p,0});
s.insert((node){r+1,now.r,now.root,0});
}
else
{
split(p,now.root,1,n,now.r-r);
s.insert((node){now.l,r,now.root,1});
s.insert((node){r+1,now.r,p,1});
}
}
int p=0;
while(2333)
{
it=s.lower_bound((node){0,l,0,0});
if(it==s.end()||(it->l)>r)break;
node now=*it;s.erase(it);
p=merge(p,now.root);
}
return p;
}
int main()
{
//freopen("test.in","r",stdin);
//freopen("test.out","w",stdout);
n=read(),m=read();
for(int i=1;i<=n;i++)a[i]=read();
for(int i=1;i<=n;i++)
{
int p=0;
insert(p,1,n,a[i]);s.insert((node){i,i,p,0});
}
for(int i=1;i<=m;i++)
{
int op=read(),l=read(),r=read(),p;
p=nodesplit(l,r);
s.insert((node){l,r,p,op});
}
Q=read();
int p=nodesplit(Q,Q);
printf("%d",find(p,1,n,1));
return 0;
}

luoguP2824 [HEOI2016/TJOI2016]排序(线段树分裂做法)的更多相关文章

  1. [HEOI2016/TJOI2016]排序 线段树+二分

    [HEOI2016/TJOI2016]排序 内存限制:256 MiB 时间限制:6000 ms 标准输入输出 题目类型:传统 评测方式:文本比较 题目描述 在2016年,佳媛姐姐喜欢上了数字序列.因而 ...

  2. [Luogu P2824] [HEOI2016/TJOI2016]排序 (线段树+二分答案)

    题面 传送门:https://www.luogu.org/problemnew/show/P2824 Solution 这题极其巧妙. 首先,如果直接做m次排序,显然会T得起飞. 注意一点:我们只需要 ...

  3. BZOJ.4552.[HEOI2016/TJOI2016]排序(线段树合并/二分 线段树)

    题目链接 对于序列上每一段连续区间的数我们都可以动态开点建一棵值域线段树.初始时就是\(n\)棵. 对于每次操作,我们可以将\([l,r]\)的数分别从之前它所属的若干段区间中分离出来,合并. 对于升 ...

  4. 洛谷$P2824\ [HEOI2016/TJOI2016]$ 排序 线段树+二分

    正解:线段树+二分 解题报告: 传送门$QwQ$ 昂着题好神噢我$jio$得$QwQQQQQ$,,, 开始看到长得很像之前考试题的亚子,,,然后仔细康康发现不一样昂$kk$,就这里范围是$[1,n]$ ...

  5. Luogu P2824 [HEOI2016/TJOI2016]排序 线段树+脑子

    只会两个$log$的$qwq$ 我们二分答案:设答案为$ans$,则我们把$a[i]<=ans$全部设成$0$,把$a[i]>ans$全部设成$1$,扔到线段树里,这样区间排序(升序)就是 ...

  6. day 1 晚上 P2824 [HEOI2016/TJOI2016]排序 线段树

    #include<iostream> #include<cstdio> #include<cstdlib> #include<cmath> #inclu ...

  7. 排序HEOI2016/TJOI2016 二分+线段树判定

    LINK:排序 此题甚好我一点思路都没有要是我当时省选此题除了模拟我恐怕想不到还可以二分 还可以线段树... 有点ex 不太好写 考虑 暴力显然每次给出询问我们都是可以直接sort的 无视地形无视一切 ...

  8. luoguP2824 [HEOI2016/TJOI2016]排序(二分答案做法)

    题意 这题的思路实在巧妙. 首先我们肯定无法对区间进行sort,那么考虑如何使得sort简化. 问:如果给的序列是一个0-1序列,让你区间排序,那么怎么做? 答:建一颗线段树维护sum,求出当前区间中 ...

  9. BZOJ4552 HEOI/TJOI2016 排序 线段树、二分答案

    题目传送门:https://www.lydsy.com/JudgeOnline/problem.php?id=4552 题意:给出一个$1$到$N$的全排列,对其进行$M$次排序,每次排序将区间$[l ...

随机推荐

  1. Day2 - Python基础2 列表、字符串、字典、集合、文件、字符编码

    本节内容 列表.元组操作 数字操作 字符串操作 字典操作 集合操作 文件操作 字符编码与转码 1. 列表.元组操作 列表是我们最以后最常用的数据类型之一,通过列表可以对数据实现最方便的存储.修改等操作 ...

  2. python 格式化打印

    #coding=utf-8 import time; start_time = time.time()for a in range(0,1001): for b in range(0,1001): f ...

  3. Java之封装性

    封装概述 面向对象编程语言是对客观世界的模拟,客观世界里成员变量都是隐藏在对象内部的,外界无法直接操作和修改. 封装可以被认为是一个保护屏障,防止该类的代码和数据被其他类随意访问.要访问该类的数据,必 ...

  4. <Stack> (高频)394 ( 高频)224

    394. Decode String 四种情况: 1. 数字,把之前有的数字乘以10再加本数字 2. ' [ ', 入口, 把之前的数字压入栈中并num归零. 3. ' ] ' ,出口,归零.用dfs ...

  5. postgresql 笔记

    客户端GUI 在官网下载一个,在安装的时候,不安装 server 端,会在客户端 安装一个 pgadmin .

  6. Fuzzy finder(fzf+vim) 使用入门指南

    今天无意中尝试了fzf,才发现这个工具的威力无穷,毕竟是非常好的工具,第一次都把它的优点都释放出来也不现实,先熟悉一下吧,后面在实战中再不断地学习总结. 它是什么: Fuzzy finder 是一款使 ...

  7. 瓜子IM智能客服系统的数据架构设计(整理自现场演讲)

    本文由ITPub根据封宇在[第十届中国系统架构师大会(SACC2018)]现场演讲内容整理而成. 1.引言 瓜子业务重线下,用户网上看车.预约到店.成交等许多环节都发生在线下.瓜子IM智能客服系统的目 ...

  8. 《细说PHP》 第四版 样章 第二章 PHP的应用与发展 5

    2.5  如何学习PHP PHP以其简单易学的特点,以及敏捷开发的优势,从一个几乎不被人知的开源项目,慢慢成长为技术人员首选的动态Web设计工具,与其他语言相比,PHP表现得更好.更快.更简单易学.尽 ...

  9. IT兄弟连 Java语法教程 数据类型1

    Java是强类型化的语言 Java是一种强类型话的语言,在开始时指出这一点是很重要的.实际上,Java的安全性和健壮性正是部分来自这一事实.强类型意味着什么呢?首先,每个变量都具有一种类型,每个表达式 ...

  10. 使用pymysql模块进行封装,自动化不可或缺的数据库校验

    import pymysql class HandleMysql: ''' 定义一个mysql处理类 ''' def __init__(self, hostname, username, passwo ...