关闭

[bzoj4552][TJOI&HEOI2016]排序

1987人阅读 评论(2) 收藏 举报
分类:

题目大意

有一个n的排列,进行m次操作,每次操作是将一个区间升序或降序排序。
请你输出m次操作后第p个位置的值。

二分答案

题解好机智!
我们二分答案x,然后就是判断a[p]>=x?
把原序列转化为01序列,0表示小于x,1表示大于等于x。
那么区间升序排序其实就是把0全放前面,1都放后面。
用线段树兹瓷区间赋值就好了。
然后只需要维护区间0的个数。

#include<cstdio>
#include<algorithm>
#define fo(i,a,b) for(i=a;i<=b;i++)
using namespace std;
const int maxn=100000+10;
int a[maxn],sum[maxn*5],set[maxn*5],ask[maxn][3];
bool bz[maxn*5];
int i,j,k,l,r,mid,t,n,m,p;
void mark(int p,int l,int r,int v){
    sum[p]=(v?0:r-l+1);
    set[p]=v;
    bz[p]=1;
}
void down(int p,int l,int r){
    int mid=(l+r)/2;
    if (bz[p]){
        mark(p*2,l,mid,set[p]);
        mark(p*2+1,mid+1,r,set[p]);
        bz[p]=0;
    }
}
void change(int p,int l,int r,int a,int b,int v){
    if (a>b) return;
    if (l==a&&r==b){
        mark(p,l,r,v);
        return;
    }
    down(p,l,r);
    int mid=(l+r)/2;
    if (b<=mid) change(p*2,l,mid,a,b,v);
    else if (a>mid) change(p*2+1,mid+1,r,a,b,v);
    else change(p*2,l,mid,a,mid,v),change(p*2+1,mid+1,r,mid+1,b,v);
    sum[p]=sum[p*2]+sum[p*2+1];
}
int query(int p,int l,int r,int a,int b){
    if (l==a&&r==b) return sum[p];
    down(p,l,r);
    int mid=(l+r)/2;
    if (b<=mid) return query(p*2,l,mid,a,b);
    else if (a>mid) return query(p*2+1,mid+1,r,a,b);
    else return query(p*2,l,mid,a,mid)+query(p*2+1,mid+1,r,mid+1,b);
}
bool check(int x){
    fo(i,1,n) 
        if (a[i]<x) change(1,1,n,i,i,0);else change(1,1,n,i,i,1);
    fo(i,1,m){
        t=query(1,1,n,ask[i][1],ask[i][2]);
        if (ask[i][0]){
            change(1,1,n,ask[i][1],ask[i][2]-t,1);
            change(1,1,n,ask[i][2]-t+1,ask[i][2],0);
        }
        else{
            change(1,1,n,ask[i][1],ask[i][1]+t-1,0);
            change(1,1,n,ask[i][1]+t,ask[i][2],1);
        }
    }
    return !query(1,1,n,p,p);
}
int main(){
    scanf("%d%d",&n,&m);
    fo(i,1,n) scanf("%d",&a[i]);
    fo(i,1,m) scanf("%d%d%d",&ask[i][0],&ask[i][1],&ask[i][2]);
    scanf("%d",&p);
    l=1;r=n;
    while (l<r){
        mid=(l+r+1)/2;
        if (check(mid)) l=mid;else r=mid-1;
    }
    printf("%d\n",l);
}
1
0
查看评论
发表评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场

[Treap套权值线段树 线段树分裂与合并] BZOJ 4552 [Tjoi2016&Heoi2016]排序

线段树合并写了不少 分裂是第一次 直接每一个有序区间用一棵权值线段树维护有哪些数 外层用treap维护顺序 然后排序就把代表这段的很多颗线段树合并在一起 两端处会割开某个有序区间 会涉及分裂操作...
  • u014609452
  • u014609452
  • 2017-01-30 19:34
  • 464

BZOJ 4552 排序(二分 || 线段树合并)

题意:对一个1~n的全排列, 进行m次局部排序 (0, l, r)将区间[l, r]数字升序排列 (1, l, r)将区间[l, r]数字降序排列 仅一次询问, 询问m次局部排序后第q位置上...
  • CillyB
  • CillyB
  • 2017-08-12 00:59
  • 202

【Tjoi2016&Heoi2016】排序

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

[Treap套权值线段树 线段树分裂与合并] BZOJ 4552 [Tjoi2016&Heoi2016]排序

线段树合并写了不少 分裂是第一次 直接每一个有序区间用一棵权值线段树维护有哪些数 外层用treap维护顺序 然后排序就把代表这段的很多颗线段树合并在一起 两端处会割开某个有序区间 会涉及分裂操作...
  • u014609452
  • u014609452
  • 2017-01-30 19:34
  • 464

Java选择排序(升序和降序)

选择排序是简单排序的一种,其排序思想为:首先将第一个数标记为最大数,其位置为最大数的位置;然后排除第一个数,使用第一个数和剩下的数依次比较,若剩下的数大于第一个数,则继续比较,直到找到最大数为止;最后...
  • shenjichao2008
  • shenjichao2008
  • 2014-03-21 16:22
  • 2166

C#_List<T>升序排序和降序排序

using System; using System.Collections; using System.Collections.Generic; using System.IO; using Sys...
  • c201038795050
  • c201038795050
  • 2015-12-27 15:06
  • 6497

C++ sort 排序(降序、升序)使用总结

一、升序     C++ sort 函数十分方便,可以对内置类型也可对自定义类型进行快速排序,内置类型的使用比较简单,下面主要讨论自定义类型的排序,一般有如下几种使用方法: 1.1 重载比较操作符...
  • zhangpiu
  • zhangpiu
  • 2016-01-22 20:48
  • 8825

Java 二维数组根据维度升序或降序排序

/*二维数组排序*/ public static String[][] arraySort( String array[][], int row, boolean sort,int len) { ...
  • romaticjun2011
  • romaticjun2011
  • 2014-10-16 13:51
  • 3857

【bzoj4552】【Tjoi2016&Heoi2016】【排序】【线段树】

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

js实现表格table的升序降序功能

自定义表格排序(JS) (function($){ //插件 $.extend($,{ //命名空间 sortTable:{ ...
  • u014801403
  • u014801403
  • 2016-12-07 08:59
  • 2200
    个人资料
    • 访问:327513次
    • 积分:11636
    • 等级:
    • 排名:第1512名
    • 原创:814篇
    • 转载:4篇
    • 译文:0篇
    • 评论:201条
    最新评论
    文章分类