luoguP2824 [HEOI2016/TJOI2016]排序(线段树分裂做法)
题意
所谓线段树分裂其实是本题的在线做法。
考虑如果我们有一个已经排好序的区间的权值线段树,那么就可以通过线段树上二分的方法得到第\(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]排序(线段树分裂做法)的更多相关文章
- [HEOI2016/TJOI2016]排序 线段树+二分
[HEOI2016/TJOI2016]排序 内存限制:256 MiB 时间限制:6000 ms 标准输入输出 题目类型:传统 评测方式:文本比较 题目描述 在2016年,佳媛姐姐喜欢上了数字序列.因而 ...
- [Luogu P2824] [HEOI2016/TJOI2016]排序 (线段树+二分答案)
题面 传送门:https://www.luogu.org/problemnew/show/P2824 Solution 这题极其巧妙. 首先,如果直接做m次排序,显然会T得起飞. 注意一点:我们只需要 ...
- BZOJ.4552.[HEOI2016/TJOI2016]排序(线段树合并/二分 线段树)
题目链接 对于序列上每一段连续区间的数我们都可以动态开点建一棵值域线段树.初始时就是\(n\)棵. 对于每次操作,我们可以将\([l,r]\)的数分别从之前它所属的若干段区间中分离出来,合并. 对于升 ...
- 洛谷$P2824\ [HEOI2016/TJOI2016]$ 排序 线段树+二分
正解:线段树+二分 解题报告: 传送门$QwQ$ 昂着题好神噢我$jio$得$QwQQQQQ$,,, 开始看到长得很像之前考试题的亚子,,,然后仔细康康发现不一样昂$kk$,就这里范围是$[1,n]$ ...
- Luogu P2824 [HEOI2016/TJOI2016]排序 线段树+脑子
只会两个$log$的$qwq$ 我们二分答案:设答案为$ans$,则我们把$a[i]<=ans$全部设成$0$,把$a[i]>ans$全部设成$1$,扔到线段树里,这样区间排序(升序)就是 ...
- day 1 晚上 P2824 [HEOI2016/TJOI2016]排序 线段树
#include<iostream> #include<cstdio> #include<cstdlib> #include<cmath> #inclu ...
- 排序HEOI2016/TJOI2016 二分+线段树判定
LINK:排序 此题甚好我一点思路都没有要是我当时省选此题除了模拟我恐怕想不到还可以二分 还可以线段树... 有点ex 不太好写 考虑 暴力显然每次给出询问我们都是可以直接sort的 无视地形无视一切 ...
- luoguP2824 [HEOI2016/TJOI2016]排序(二分答案做法)
题意 这题的思路实在巧妙. 首先我们肯定无法对区间进行sort,那么考虑如何使得sort简化. 问:如果给的序列是一个0-1序列,让你区间排序,那么怎么做? 答:建一颗线段树维护sum,求出当前区间中 ...
- BZOJ4552 HEOI/TJOI2016 排序 线段树、二分答案
题目传送门:https://www.lydsy.com/JudgeOnline/problem.php?id=4552 题意:给出一个$1$到$N$的全排列,对其进行$M$次排序,每次排序将区间$[l ...
随机推荐
- SpringCloud学习笔记(二、SpringCloud Config)
目录: 配置中心简介 SpringCloud Config服务端 SpringCloud Config客户端 动态配置属性bean 一些补充(源码分析):Spring事件监听.健康检查health() ...
- day83_11_1 阿里配python使用。
一.环境准备. 1.首先需要在支付包中注册开发者模式,并注册沙箱,模拟支付过程. https://openhome.alipay.com/platform/appDaily.htm?tab=info ...
- SP1043 GSS1 - Can you answer these queries I 线段树
问题描述 LG-SP1043 题解 GSS 系列第一题. \(q\) 个询问,求 \([x,y]\) 的最大字段和. 线段树,维护 \([x,y]\) 的 \(lmax,rmax,sum,val\) ...
- As Simple as One and Two
time limit per test3 secondsmemory limit per test256 megabytesinput: standard inputoutput: standard ...
- java启动参数
java命令启动应用所使用的参数,基本是用于JVM的,某种程度上也叫做JVM参数.总的来说,java启动参数共分为三大类,分别是: 标准参数(-):相对稳定的参数,每个版本的JVM都可用. 非标准X参 ...
- php 压缩字符串
压缩字符串: base64_encode(gzcompress(serialize($data))) 解压字符串: unserialize(gzuncompress(base64_decode($se ...
- JS 从内存空间谈到垃圾回收机制
壹 ❀ 引 从事计算机相关技术工作的同学,对于内存空间相关概念多少有所耳闻,毕竟像我这种非计算机科班出身的人,对于栈堆,垃圾回收都能简单说道几句:当我明白JS 基本类型与引用类型数据存储方式不同,才 ...
- 详解JAVA8Stream API {全}
1: 概述 1.1 优势 1.2 与传统迭代器的区分 1.3 流的操作类型分为两种: 2:流的构造与转换 2:1 常见构造 2.2: 三大包装类型的构造 2.3 并行流的规则输出 2.4 流的转换 3 ...
- leaflet-webpack 入门开发系列一初探篇(附源码下载)
前言 leaflet-webpack 入门开发系列环境知识点了解: node 安装包下载webpack 打包管理工具需要依赖 node 环境,所以 node 安装包必须安装,上面链接是官网下载地址 w ...
- CodeWarrior IDE烧写介绍
点击Flash烧写 选择芯片系列 下面将以PPC8548 NOR Flash烧写为例 默认配置文件目录:C:\Program Files (x86)\Freescale\CodeWarrior PA ...