Description

给出一个$1$到$n$的全排列,现在对这个全排列序列进行$m$次局部排序,排序分为$2$种:

$1.(0,l,r)$表示将区间$[l,r]$的数字升序排序;

$2.(1,l,r)$表示将区间$[l,r]$的数字降序排序.

最后询问第$q$位置上的数字.

Input

第$1$行为两个整数$n$和$m$.$n$表示序列的长度,$m$表示局部排序的次数.

第$2$行为$n$个整数,表示$1$到$n$的一个全排列.

接下来输入$m$行,每$1$行有$3$个整数$op,l,r$.

$op$为$0$代表升序排序,$op$为$1$代表降序排序;$l,r$表示排序的区间.

最后输入一个整数$q$,$q$表示排序完之后询问的位置.

Output

输出数据仅有一行一个整数,表示按照顺序将全部的部分排序结束后第$q$位置上的数字.

Sample Input

6 3
1 6 2 5 3 4
0 1 4
1 3 6
0 2 4
3

Sample Output

5

HINT

$1\;\leq\;n,m\;\leq\;10^5,1\;\leq\;q\;\leq\;n.$

Solution

二分答案$ans$.

把$\;\leq\;ans$的位置标为$1$,其余标为$0$.

线段树维护排序操作,最后判断位置$q$上是否为$0$.

#include<cmath>
#include<ctime>
#include<queue>
#include<stack>
#include<cstdio>
#include<vector>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define N 100005
#define M 300005
using namespace std;
struct linetree{
int l,r,t,op;
}lt[M];
struct quest{
int op,l,r;
}b[N];
int a[N],n,m,q,t,l,r,mid;
inline void build(int u,int l,int r,int k){
lt[u].l=l;lt[u].r=r;lt[u].op=0;
if(lt[u].l<lt[u].r){
int lef=u<<1,rig=u<<1|1;
int mid=(lt[u].l+lt[u].r)>>1;
build(lef,l,mid,k);build(rig,mid+1,r,k);
lt[u].t=lt[lef].t+lt[rig].t;
}
else if(a[lt[u].l]>=k) lt[u].t=0;
else lt[u].t=1;
}
inline void cover(int u,int i){
if(lt[u].l>=b[i].l&&lt[u].r<=b[i].r){
if(!b[i].op){
lt[u].op=-1;
if(lt[u].l-b[i].l+1<=t){
lt[u].t=min(t-(lt[u].l-b[i].l),lt[u].r-lt[u].l+1);
}
else lt[u].t=0;
}
else{
lt[u].op=1;
if(b[i].r-lt[u].r+1<=t){
lt[u].t=min(t-(b[i].r-lt[u].r),lt[u].r-lt[u].l+1);
}
else lt[u].t=0;
}
}
else if(lt[u].l<lt[u].r){
int lef=u<<1,rig=u<<1|1;
int mid=(lt[u].l+lt[u].r)>>1;
if(lt[u].op>0)/*降序*/{
lt[u].op=0;lt[lef].op=lt[rig].op=1;
lt[rig].t=min(lt[u].t,lt[rig].r-lt[rig].l+1);
lt[lef].t=lt[u].t-lt[rig].t;
}
else if(lt[u].op<0)/*升序*/{
lt[u].op=0;lt[lef].op=lt[rig].op=-1;
lt[lef].t=min(lt[u].t,lt[lef].r-lt[lef].l+1);
lt[rig].t=lt[u].t-lt[lef].t;
}
if(b[i].l<=mid) cover(lef,i);
if(b[i].r>mid) cover(rig,i);
lt[u].t=lt[lef].t+lt[rig].t;
}
}
inline int ask(int u,int l,int r){
if(lt[u].l>=l&&lt[u].r<=r)
return lt[u].t;
if(lt[u].l<lt[u].r){
int lef=u<<1,rig=u<<1|1,ret=0;
int mid=(lt[u].l+lt[u].r)>>1;
if(lt[u].op>0)/*降序*/{
lt[u].op=0;lt[lef].op=lt[rig].op=1;
lt[rig].t=min(lt[u].t,lt[rig].r-lt[rig].l+1);
lt[lef].t=lt[u].t-lt[rig].t;
}
else if(lt[u].op<0)/*升序*/{
lt[u].op=0;lt[lef].op=lt[rig].op=-1;
lt[lef].t=min(lt[u].t,lt[lef].r-lt[lef].l+1);
lt[rig].t=lt[u].t-lt[lef].t;
}
if(l<=mid) ret+=ask(lef,l,r);
if(r>mid) ret+=ask(rig,l,r);
return ret;
}
}
inline bool chk(int k){
build(1,1,n,k);
for(int i=1;i<=m;++i){
t=ask(1,b[i].l,b[i].r);cover(1,i);
}
return !ask(1,q,q);
}
inline void init(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;++i)
scanf("%d",&a[i]);
for(int i=1;i<=m;++i)
scanf("%d%d%d",&b[i].op,&b[i].l,&b[i].r);
scanf("%d",&q);
l=1;r=n;
while(l<r){
mid=(l+r+1)>>1;
if(chk(mid)) l=mid;
else r=mid-1;
}
printf("%d\n",l);
}
int main(){
freopen("sort.in","r",stdin);
freopen("sort.out","w",stdout);
init();
fclose(stdin);
fclose(stdout);
return 0;
}

[bzoj4552][Tjoi2016][Heoi2016]排序的更多相关文章

  1. bzoj千题计划128:bzoj4552: [Tjoi2016&Heoi2016]排序

    http://www.lydsy.com/JudgeOnline/problem.php?id=4552 二分答案 把>=mid 的数看做1,<mid 的数看做0 这样升序.降序排列相当于 ...

  2. [BZOJ4552][TJOI2016&&HEOI2016]排序(二分答案+线段树/线段树分裂与合并)

    解法一:二分答案+线段树 首先我们知道,对于一个01序列排序,用线段树维护的话可以做到单次排序复杂度仅为log级别. 这道题只有一个询问,所以离线没有意义,而一个询问让我们很自然的想到二分答案.先二分 ...

  3. BZOJ4552 [Tjoi2016&Heoi2016]排序 【二分 + 线段树】

    题目链接 BZOJ4552 题解 之前去雅礼培训做过一道题,\(O(nlogn)\)维护区间排序并能在线查询 可惜我至今不能get 但这道题有着\(O(nlog^2n)\)的离线算法 我们看到询问只有 ...

  4. BZOJ4552: [Tjoi2016&Heoi2016]排序

    Description 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题 ,需要你来帮助他.这个难题是这样子的:给出一个1到n的全排列,现在对这 ...

  5. [bzoj4552][Tjoi2016&Heoi2016]排序-二分+线段树

    Brief Description DZY有一个数列a[1..n],它是1∼n这n个正整数的一个排列. 现在他想支持两种操作: 0, l, r: 将a[l..r]原地升序排序. 1, l, r: 将a ...

  6. BZOJ4552:[TJOI2016&HEOI2016]排序(线段树,二分)

    Description 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题,需要你来帮助他. 这个难题是这样子的:给出一个1到n的全排列,现在对这 ...

  7. 2018.08.01 BZOJ4552: [Tjoi2016&Heoi2016]排序(二分+线段树)

    传送门 线段树简单题. 二分答案+线段树排序. 实际上就是二分答案mid" role="presentation" style="position: relat ...

  8. BZOJ4552 Tjoi2016&Heoi2016排序 【二分+线段树】*

    Description 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题,需要你来帮助他.这个难题是这样子的:给出一个1到n的全排列,现在对这个 ...

  9. [BZOJ4552][Tjoi2016&Heoi2016]排序(二分答案+线段树)

    二分答案mid,将>=mid的设为1,<mid的设为0,这样排序就变成了区间修改的操作,维护一下区间和即可 然后询问第q个位置的值,为1说明>=mid,以上 时间复杂度O(nlog2 ...

随机推荐

  1. PL/SQL Developer 11 64bit 安装和配置

    安装后, 1. 解压 instant client 到plsql developer 的安装目录    注意, 此版本只支持 instantclient_11_x, 不支持 instantclient ...

  2. Spring的反射机制和依赖注入

    我们知道,Spring中大量使用了反射机制,那么究竟是什么地方使用了呢? spring的一大核心概念是注入, 但是,这存在的一个前提就是类是由spring管理起来的. 反射是根据className生成 ...

  3. Java设计模式之-----策略模式

    首先,我们来看下策略模式的概念.一般的解释如下:     策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换.策略模式让算法独立于使用它的客户而独立变化.(原文:The St ...

  4. Linux 网络编程详解十一

    /** * read_timeout - 读超时检测函数,不含读操作 * @fd:文件描述符 * @wait_seconds:等待超时秒数,如果为0表示不检测超时 * 成功返回0,失败返回-1,超时返 ...

  5. QT 网络编程三(TCP版)

    QT客户端 //widget.h #ifndef WIDGET_H #define WIDGET_H #include <QWidget> #include <QTcpSocket& ...

  6. 学习C++.Primer.Plus 4 复合类型

    本章介绍的有复合类型有: 数组. 字符串. 结构. 共用体. 指针 数组: 声明数组时数组长度必须为常量(或const). 只有初始化时可以用“=”,其它时候均不可以给数组直接赋值,除了赋值的元素以外 ...

  7. SQL基础之数据库快照

    1.认识快照 如名字一样,数据库快照就可以理解为数据库某一时刻的照片,它记录了此时数据库的数据信息.如果要认识快照的本质,那就要了解快照的工作原理.当我们执行t-sql创建快照后,此时就会创建一个或多 ...

  8. 知乎日报win10版 - 天天读报【开源】

    业余时间写的一个知乎日报win10版客户端,支持收藏,评论,点赞等. 商店地址:https://www.microsoft.com/zh-cn/store/apps/%E5%A4%A9%E5%A4%A ...

  9. MVC架构设计——EF-Code First

    详情参考:http://www.cnblogs.com/guomingfeng/archive/2013/05/28/mvc-ef-repository.html

  10. No Launcher activity found!

    已经研究Android有几天了,刚开始写的代码说安装成功,但是在AVD没有显示.左看代码,右看代码,总是没找到错误, <application android:allowBackup=" ...