https://www.lydsy.com/JudgeOnline/problem.php?id=4552

https://blog.csdn.net/zawedx/article/details/51818475

区间排序,这道题需要写两个线段树还要维护一个链表,有些细节,对我目前的代码能力来说有点算是码农题,但是理解思路之后调试起来出乎意料地简单。

这个写法的复杂度据说是nlogn的,我也不是很会算,反正能过就行(bushi)。

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
using namespace std;
#define LL long long
#define lc x*2
#define rc x*2+1
#define mid (l+r)/2
const int maxn=;
const int maxm=;
int n,m;
struct sgtree{
int rn[maxm];
void updata(int x){rn[x]=(rn[rc]?rn[rc]:rn[lc]);}
void ins(int x,int l,int r,int k,int v){
if(l==r){rn[x]=v;return;}
if(k>mid)ins(rc,mid+,r,k,v);
else ins(lc,l,mid,k,v);
updata(x);
}
int ask(int x,int l,int r,int k){
if(!rn[x])return ;
if(l==r)return rn[x];
int zz=;
if(mid<k)zz=ask(rc,mid+,r,k);
if(zz)return zz;
if(mid<=k)return rn[lc];
else return ask(lc,l,mid,k);
}
}e;
struct nod{ int l,r,tp,nxt; }c[maxm];
struct node{ int l,r,sum; }t[maxm];
queue<int>q;
inline int newp(){
int z=q.front();q.pop();
return z;
}
inline void delp(int x){
q.push(x);t[x]=t[];
}
void build(int x,int l,int r,int k){
t[x].sum=;
if(l==r)return;
if(k>mid)build(t[x].r=newp(),mid+,r,k);
else build(t[x].l=newp(),l,mid,k);
}
int mmerg(int x,int y){
if((!x)||(!y))return x+y;
t[x].l=mmerg(t[x].l,t[y].l);
t[x].r=mmerg(t[x].r,t[y].r);
t[x].sum=t[x].sum+t[y].sum;
delp(y); return x;
}
void mspli(int x,int y,int k){
int z=t[t[x].l].sum;
if(z<k)mspli(t[x].r,t[y].r=newp(),k-z);else swap(t[x].r,t[y].r);
if(z>k)mspli(t[x].l,t[y].l=newp(),k);
t[y].sum=t[x].sum-k;
t[x].sum=k;
}
int mask(int x,int l,int r,int k){
if(l==r)return l;
int z=t[t[x].l].sum;
if(k>z)return mask(t[x].r,mid+,r,k-z);
else return mask(t[x].l,l,mid,k);
}
int main(){
int x,v,op,l,r;
for(int i=;i<=maxm-;++i)q.push(i);
scanf("%d%d",&n,&m);
for(int i=,las=maxm-;i<=n;++i,las=x){
scanf("%d",&v);
c[las].nxt=x=newp();
c[x]=(nod){i,i,,};
e.ins(,,n,i,x);build(x,,n,v);
}
for(int i=;i<=m;++i){
scanf("%d%d%d",&op,&l,&r);
/*for(x=1;x<=n;++x){
int w=e.ask(1,1,n,x);
//cout<<w<<' ';
if(c[w].tp)printf("%d ",mask(w,1,n,c[w].r-x+1));
else printf("%d ",mask(w,1,n,x-c[w].l+1));
}printf("\n");*/
int lef=e.ask(,,n,l),rig,now,nxt;
if(l==c[lef].l){
rig=newp();
swap(t[rig],t[lef]); swap(c[rig],c[lef]);
c[now=lef]=(nod){l,r,op,rig};
}
else{
c[now=newp()]=(nod){l,r,op,rig=newp()};
if(!c[lef].tp) mspli(lef,rig,l-c[lef].l);
else {mspli(lef,rig,c[lef].r-l+);swap(t[lef],t[rig]);}
c[rig]=c[lef];c[rig].l=l;
c[lef].r=l-;c[lef].nxt=now;
}
for(nxt=rig;nxt&&r>=c[nxt].r;){
//cout<<t[nxt].sum<<endl;
mmerg(now,nxt);e.ins(,,n,c[nxt].l,);
int qq=nxt;nxt=c[nxt].nxt;c[qq]=c[];
}
c[now].nxt=nxt;
if(nxt!=&&c[nxt].l<=r){
e.ins(,,n,c[nxt].l,);
int zz=newp();
if(!c[nxt].tp) {mspli(nxt,zz,r-c[nxt].l+);swap(t[nxt],t[zz]);}
else mspli(nxt,zz,c[nxt].r-r);
mmerg(now,zz);
c[nxt].l=r+;
e.ins(,,n,r+,nxt);
}
e.ins(,,n,l,now);
}
/*for(x=1;x<=n;++x){
int w=e.ask(1,1,n,x);
//cout<<w<<' ';
if(c[w].tp)printf("%d ",mask(w,1,n,c[w].r-x+1));
else printf("%d ",mask(w,1,n,x-c[w].l+1));
}printf("\n");*/
scanf("%d",&x);
int w=e.ask(,,n,x);
if(c[w].tp)printf("%d\n",mask(w,,n,c[w].r-x+));
else printf("%d\n",mask(w,,n,x-c[w].l+));
return ;
}

BZOJ 4552 [Tjoi2016&Heoi2016]排序 线段树的分裂和合并的更多相关文章

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

    目录 此代码是个假代码,只能糊弄luogu,以后再改,路过大佬也可以帮一下辣 update 10.6 此代码是个假代码,只能糊弄luogu,以后再改,路过大佬也可以帮一下辣 /* //fang zhi ...

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

    听说是BC原题. 好题,二分答案变成01序列,就可以方便的用线段树维护了. 然后就是区间查询和覆盖了. #include <map> #include <cmath> #inc ...

  3. BZOJ 4552: [Tjoi2016&Heoi2016]排序

    4552: [Tjoi2016&Heoi2016]排序 Time Limit: 60 Sec  Memory Limit: 256 MBSubmit: 579  Solved: 322[Sub ...

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

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

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

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

  6. bzoj 4552: [Tjoi2016&Heoi2016]排序——二分+线段树

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

  7. bzoj 4552: [Tjoi2016&Heoi2016]排序【二分+线段树】

    二分值mid,然后把>=mid的赋值为1,其他赋值为0,每次排序就是算出区间内01的个数,然后分别把0和1放到连续的一段内,这些都可以用线段树来维护 二分的判断条件是操作完之后q位置上是否为1 ...

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

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

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

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

随机推荐

  1. mysql binglog server的设置方法【原创】

    MySQL备份数据都是MySQL备份+binlog,这样才能保证数据的完整性.下面就是利用mysqlbinlog搭建mysql binlog server,可以把binlog传到远程存储上. 试验环境 ...

  2. PNG,JPEG,BMP,JIF图片格式详解及其对比

    原文地址:http://blog.csdn.net/u012611878/article/details/52215985 图片格式详解 不知道大家有没有注意过网页里,手机里,平板里的图片,事实上,图 ...

  3. dell R720服务器设置开机启动顺序

    开机按F2进入系统启动设置,也可以按F11进入快速启动配置

  4. MonkeyRunner之MonkeyRecorder录制回放脚本(亲测可正常运行)

    MonkeyRunner可以录制和回放脚本 前置条件: 电脑连接手机,输入adb devices 看看返回是否手机设备列表(我是真机,模拟器也可以) 配置好安卓sdk和Python环境 step: 1 ...

  5. PYTHON-流程控制之if/while/for-练习

    # 1 练习题## 简述编译型与解释型语言的区别,且分别列出你知道的哪些语言属于编译型,哪些属于解释型# 编译型:C, 谷歌翻译,一次翻译后结果后重复使用# 解释型:Python, 同声传译,边执行边 ...

  6. vue系列之MVVM框架

    当数据发生变化时,ViewModel就会检测到,然后通知相应的View改变 当用户操作View时,ViewModel就会检测到,然后Model,修改相应的数据,最终实现双向绑定 适用场景:针对具有复杂 ...

  7. TestNG测试方法

    @Test(enabled = false)有助于禁用此测试用例. 分组测试是TestNG中的一个新的创新功能,使用<groups>标记在testng.xml文件中指定分组. 它可以在&l ...

  8. LeetCode(54):螺旋矩阵

    Medium! 题目描述: 给定一个包含 m x n 个元素的矩阵(m 行, n 列),请按照顺时针螺旋顺序,返回矩阵中的所有元素. 示例 1: 输入: [ [ 1, 2, 3 ], [ 4, 5, ...

  9. Laravel API 限速异常 HTTPException Too Many Attemps

    最近线上 Laravel 错误日志发现了一个异常 HTTPException Too Many Attemps 而我一个同事在本地开发的过程中也频繁遇到这个异常.测试环境:Laravel 5.5. 网 ...

  10. php 爬取数据

    简单. 灵活.强大的PHP采集工具,让采集更简单一点. 简介: QueryList使用jQuery选择器来做采集,让你告别复杂的正则表达式:QueryList具有jQuery一样的DOM操作能力.Ht ...