解法一:二分答案+线段树

首先我们知道,对于一个01序列排序,用线段树维护的话可以做到单次排序复杂度仅为log级别。

这道题只有一个询问,所以离线没有意义,而一个询问让我们很自然的想到二分答案。先二分出这个位置上的数是多少,然后将所有小于等于的数全部赋为0,其余赋为1,这样每次排序都是01序列排序了。如果最后p位置上的数为0则说明最终答案小于等于当前二分的答案,反之亦然。

这样这个问题就在$O(n \log^2 n)$的复杂度内解决了。

 #include<cstdio>
#include<algorithm>
#define ls (x<<1)
#define rs ((x<<1)|1)
#define lson ls,L,mid
#define rson rs,mid+1,R
#define rep(i,l,r) for (int i=l; i<=r; i++)
using namespace std; const int N=;
int n,m,qry,a[N],c[N],sm[N<<],tag[N<<];
struct P{ int op,l,r; }b[N]; void push(int x,int L,int R){
if (tag[x]==-) return;
int mid=(L+R)>>;
sm[ls]=(mid-L+)*tag[x]; tag[ls]=tag[x];
sm[rs]=(R-mid)*tag[x]; tag[rs]=tag[x];
tag[x]=-;
} void build(int x,int L,int R){
tag[x]=-;
if (L==R) { sm[x]=c[L]; return; }
int mid=(L+R)>>;
build(lson); build(rson);
sm[x]=sm[ls]+sm[rs];
} void mdf(int x,int L,int R,int l,int r,int k){
if (L==l && r==R){ sm[x]=k*(R-L+); tag[x]=k; return; }
int mid=(L+R)>>; push(x,L,R);
if (r<=mid) mdf(lson,l,r,k);
else if (l>mid) mdf(rson,l,r,k);
else mdf(lson,l,mid,k),mdf(rson,mid+,r,k);
sm[x]=sm[ls]+sm[rs];
} int que(int x,int L,int R,int l,int r){
if (L==l && r==R) return sm[x];
int mid=(L+R)>>; push(x,L,R);
if (r<=mid) return que(lson,l,r);
else if (l>mid) return que(rson,l,r);
else return que(lson,l,mid)+que(rson,mid+,r);
} int main(){
scanf("%d%d",&n,&m);
rep(i,,n) scanf("%d",&a[i]);
rep(i,,m) scanf("%d%d%d",&b[i].op,&b[i].l,&b[i].r);
scanf("%d",&qry);
int L=,R=n;
while (L<R){
int mid=(L+R)>>;
rep(i,,n) if (a[i]<=mid) c[i]=; else c[i]=;
build(,,n);
rep(i,,m){
int l=b[i].l,r=b[i].r,s=que(,,n,l,r);
if (b[i].op==){
if (l<=r-s) mdf(,,n,l,r-s,);
if (r-s+<=r) mdf(,,n,r-s+,r,);
}else{
if (l<=l+s-) mdf(,,n,l,l+s-,);
if (l+s<=r) mdf(,,n,l+s,r,);
}
}
if (que(,,n,qry,qry)==) R=mid; else L=mid+;
}
printf("%d\n",L);
return ;
}

解法二:线段树分裂与合并

对每个连续的有序区间维护一棵线段树(初始时每个点都有一棵线段树)。

每次将一个区间排序时,先将跨过这个区间端点的区间分裂,再将这个区间包含的所有小区间合并。

set维护区间,区间合并通过线段树合并实现,时间复杂度为$O(n\log n)$。

可以垃圾回收,由于每次分裂只会新增log n个点,加上初始时新建的点,空间复杂度为$O((n+m)\log n)$。

这个方法不仅复杂度只有一个log,还能在最后求出整个序列而不是单个序列。

 #include<set>
#include<cstdio>
#include<algorithm>
#define lson ls[x],L,mid
#define rson rs[x],mid+1,R
#define rep(i,l,r) for (int i=(l); i<=(r); i++)
using namespace std; const int N=,M=;
int n,m,x,nd,op,l,r,top,rt[N],rev[N],stk[M],ls[M],rs[M],sz[M];
struct P{ int l,r; };
bool operator <(const P &a,const P &b){ return a.r<b.r; }
set<P>S; void del(int x){ ls[x]=rs[x]=sz[x]=; stk[++top]=x; }
int get(){ return top ? stk[top--] : ++nd; } void ins(int &x,int L,int R,int k){
if (!x) x=get(); sz[x]++;
if (L==R) return;
int mid=(L+R)>>;
if (k<=mid) ins(lson,k); else ins(rson,k);
} void split(int &x,int &y,int k,int op){
if (!x) return;
sz[y=get()]=sz[x]-k; sz[x]=k;
if (!op){
if (sz[ls[x]]==k){ rs[y]=rs[x]; rs[x]=; return; }
if (sz[ls[x]]>k){ rs[y]=rs[x]; rs[x]=; split(ls[x],ls[y],k,op); }
else split(rs[x],rs[y],k-sz[ls[x]],op);
}else{
if (sz[rs[x]]==k){ ls[y]=ls[x]; ls[x]=; return; }
if (sz[rs[x]]>k){ ls[y]=ls[x]; ls[x]=; split(rs[x],rs[y],k,op); }
else split(ls[x],ls[y],k-sz[rs[x]],op);
}
} int merge(int x,int y){
if (!x || !y) return x|y;
sz[x]+=sz[y];
ls[x]=merge(ls[x],ls[y]);
rs[x]=merge(rs[x],rs[y]);
del(y); return x;
} int que(int x,int L,int R,int k,int op){
if (L==R) return L;
int mid=(L+R)>>;
if (!op){
if (sz[ls[x]]>=k) return que(lson,k,op); else return que(rson,k-sz[ls[x]],op);
}else{
if (sz[rs[x]]>=k) return que(rson,k,op); else return que(lson,k-sz[rs[x]],op);
}
} int main(){
freopen("bzoj4552.in","r",stdin);
freopen("bzoj4552.out","w",stdout);
scanf("%d%d",&n,&m);
rep(i,,n) scanf("%d",&x),ins(rt[i],,n,x),S.insert((P){i,i});
while (m--){
scanf("%d%d%d",&op,&l,&r);
set<P>::iterator it=S.lower_bound((P){l,l});
if (it->l<l){
int L=it->l,R=it->r;
split(rt[L],rt[l],l-L,rev[L]);
S.erase(it); S.insert((P){L,l-}); S.insert((P){l,R});
rev[l]=rev[L];
}
it=S.lower_bound((P){r+,r+});
if (it!=S.end() && it->l<=r){
int L=it->l,R=it->r;
split(rt[L],rt[r+],r-L+,rev[L]);
S.erase(it); S.insert((P){L,r}); S.insert((P){r+,R});
rev[r+]=rev[L];
}
it=S.lower_bound((P){l,l}); it++;
while (it!=S.end() && it->r<=r) rt[l]=merge(rt[l],rt[it->l]),it++;
it=S.lower_bound((P){l,l});
while (it!=S.end() && it->r<=r) S.erase(it),it=S.lower_bound((P){l,l});
S.insert((P){l,r}); rev[l]=op;
}
scanf("%d",&x);
set<P>::iterator it=S.lower_bound((P){x,x});
printf("%d\n",que(rt[it->l],,n,x-(it->l)+,rev[it->l]));
return ;
}

[BZOJ4552][TJOI2016&&HEOI2016]排序(二分答案+线段树/线段树分裂与合并)的更多相关文章

  1. BZOJ 4552 [Tjoi2016&Heoi2016]排序 | 二分答案 线段树

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

  2. bzoj 4552 [Tjoi2016&Heoi2016]排序——二分答案

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4552 二分答案,把 >= mid 的设成1.< mid 的设成0,之后排序就变成 ...

  3. bzoj 4552 [Tjoi2016&Heoi2016]排序 (二分答案 线段树)

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=4552 题意: 给你一个1-n的全排列,m次操作,操作由两种:1.将[l,r]升序排序,2 ...

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

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

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

    [BZOJ4552][Tjoi2016&Heoi2016]排序 Description 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题 ...

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

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

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

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

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

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

  9. [BZOJ4556][TJOI2016&&HEOI2016]字符串(二分答案+后缀数组+RMQ+主席树)

    4556: [Tjoi2016&Heoi2016]字符串 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 1360  Solved: 545[S ...

随机推荐

  1. 【BZOJ】1040: [ZJOI2008]骑士 环套树DP

    [题意]给定n个人的ai和bi,表示第i个人能力值为ai且不能和bi同时选择,求能力值和最大的选择方案.n<=10^6. [算法]环套树DP(基环树) [题解]n个点n条边——基环森林(若干环套 ...

  2. 重载jquery on方法实现click事件在移动端的快速响应

    额,这个标题取的还真是挺装的... 其实我想表达的是jquery click事件如何在移动端自动转换成touchstart事件. 因为移动端click事件会比touchstart事件慢几拍 移动设备某 ...

  3. Codeforces 665E. Beautiful Subarrays (字典树)

    题目链接:http://codeforces.com/problemset/problem/665/E (http://www.fjutacm.com/Problem.jsp?pid=2255) 题意 ...

  4. pythonif语句和循环语句

    1.if语句用法 # if语句用法(缩进相同的成为一个代码块) score=90 if score>=60: print("合格") print("OK" ...

  5. 关于Java代码优化的35条建议

    代码优化,一个很重要的课题.可能有些人觉得没用,一些细小的地方有什么好修改的,改与不改对于代码的运行效率有什么影响呢?这个问题我是这么考虑的,就像大海里面的鲸鱼一样,它吃一条小虾米有用吗?没用,但是, ...

  6. C# 怎么显示中文格式的日期、星期几

    //该语句显示的为英文格式DateTime.Now.DayOfWeek.ToString(); //显示中文格式星期几 "星期" + DateTime.Now.ToString(& ...

  7. 如何删除git远程分支(转)

    1,在开发过程中,大家在远程创建了许多分支,有些是无用的,该如何删除呢,可以参考下面的方法. 如果不再需要某个远程分支了,比如搞定了某个特性并把它合并进了远程的 master 分支(或任何其他存放 稳 ...

  8. spring mvc整合mybaitis和log4j

    在上一篇博客中,我介绍了在mac os上用idea搭建spring mvc的maven工程,但是一个完整的项目肯定需要数据库和日志管理,下面我就介绍下spring mvc整合mybatis和log4j ...

  9. 20165301 2017-2018-2 《Java程序设计》第二周学习总结

    20165301 2017-2018-2 <Java程序设计>第二周学习总结 教材学习内容总结 第二章:基本数据类型与数组 标识符 第一个字符不能是数字 不能是关键字 不能是true.fa ...

  10. jenkins+docker持续集成实验

    在互联网时代,对于每一家公司,软件开发和发布的重要性不言而喻,目前已经形成一套标准的流程,最重要的组成部分就是持续集成(CI)及持续部署.交付(CD).本文基于Jenkins+Docker+Git实现 ...