首页 > 其他分享 >[题解]AT_abc237_g [ABC237G] Range Sort Query

[题解]AT_abc237_g [ABC237G] Range Sort Query

时间:2024-06-22 11:03:06浏览次数:7  
标签:Sort rs int 题解 mid ABC237G read 升序 inline

思路

将小于等于 \(x\) 的元素赋为 \(1\),其余的赋为 \(0\)。那么一个区间内小于等于 \(x\) 的数量就是区间中 \(1\) 的数量。

那么,将区间升序排列就是将 \(1\) 先堆在前面,将 \(0\) 堆到后面;降序排列同理。

考虑动态维护 \(x\) 的位置,记其位置为 \(t\)。如果 \(l \leq t \leq r\),则 \(t\) 可能会改变;否则不会改变。

在升序排列中,\(t\) 将会改变到最后一个 \(1\) 的位置;在降序排序中,\(t\) 将会改变到第一个 \(1\) 的位置。然后维护 \(0/1\) 可以用线段树维护。

Code

#include <bits/stdc++.h>
#define re register

using namespace std;

const int N = 2e5 + 10;
int n,q,k,x;
int arr[N];

inline int read(){
    int r = 0,w = 1;
    char c = getchar();
    while (c < '0' || c > '9'){
        if (c == '-') w = -1;
        c = getchar();
    }
    while (c >= '0' && c <= '9'){
        r = (r << 3) + (r << 1) + (c ^ 48);
        c = getchar();
    }
    return r * w;
}

struct seg{
    #define ls(u) (u << 1)
    #define rs(u) (u << 1 | 1)

    struct node{
        int l,r;
        int sum,tag;
    }tr[N << 2];
    
    inline void calc(int u,int k){
        tr[u].sum = (tr[u].r - tr[u].l + 1) * k;
        tr[u].tag = k;
    }

    inline void pushup(int u){
        tr[u].sum = tr[ls(u)].sum + tr[rs(u)].sum;
    }

    inline void pushdown(int u){
        if (~tr[u].tag){
            calc(ls(u),tr[u].tag); calc(rs(u),tr[u].tag);
            tr[u].tag = -1;
        }
    }

    inline void build(int u,int l,int r){
        tr[u] = {l,r,0,-1};
        if (l == r) return tr[u].sum = (arr[l] <= k),void();
        int mid = l + r >> 1;
        build(ls(u),l,mid); build(rs(u),mid + 1,r);
        pushup(u);
    }
    
    inline void modify(int u,int l,int r,int k){
        if (l > r) return;
        if (l <= tr[u].l && tr[u].r <= r) return calc(u,k);
        pushdown(u);
        int mid = tr[u].l + tr[u].r >> 1;
        if (l <= mid) modify(ls(u),l,r,k);
        if (r > mid) modify(rs(u),l,r,k);
        pushup(u);
    }

    inline int query(int u,int l,int r){
        if (l <= tr[u].l && tr[u].r <= r) return tr[u].sum;
        pushdown(u);
        int res = 0;
        int mid = tr[u].l + tr[u].r >> 1;
        if (l <= mid) res += query(ls(u),l,r);
        if (r > mid) res += query(rs(u),l,r);
        return res;
    }

    #undef ls
    #undef rs
}T;

int main(){
    n = read(),q = read(),k = read();
    for (re int i = 1;i <= n;i++){
        arr[i] = read();
        if (arr[i] == k) x = i;
    }
    T.build(1,1,n);
    while (q--){
        int op,l,r;
        op = read(),l = read(),r = read();
        if (op == 1){
            int a = T.query(1,l,r);
            if (l <= x && x <= r) x = l + a - 1;
            T.modify(1,l,l + a - 1,1); T.modify(1,l + a,r,0);
        }
        else{
            int a = (r - l + 1) - T.query(1,l,r);
            if (l <= x && x <= r) x = l + a;
            T.modify(1,l,l + a - 1,0); T.modify(1,l + a,r,1);
        }
    }
    printf("%d",x);
    return 0;
}

标签:Sort,rs,int,题解,mid,ABC237G,read,升序,inline
From: https://www.cnblogs.com/WaterSun/p/18261967

相关文章

  • [题解]AT_abc236_f [ABC236F] Spices
    思路首先对所有的\(c\)从小到大排序,然后对于每一个值如果之前能凑出就不选,否则就选。这样做显然是对的。令\(p_1,p_2,\dots,p_{2^n-1}\)表示将\(c\)排序之后,对应原来的下标;\(S\)表示选出数的集合;\(S'\)表示最终选出数的集合。可以证明两个问题:如果\(p_i\)可以被已选......
  • [题解]AT_abc247_f [ABC247F] Cards
    思路对于包含数\(x\)的卡牌,两张之中必定要选择一张,由此想到2-SAT的思想。我们将所有带有\(x\)的卡牌两两连边,每一条边连接的点都表示两点必须选择一个。不难发现,我们这样会得出若干个环。(因为对于每一张卡牌的出边为\(2\),一定会形成环)在每一个环中的选择情况,不会影响答......
  • [题解]AT_abc240_f [ABC240F] Sum Sum Max
    思路题目要求的是\(\max_{a=1}^{n}\{\sum_{i=1}^{a}\sum_{j=1}^{a}{A_j}\}\),所以我们将\(\sum_{i=1}^{a}\sum_{j=1}^{a}{A_j}\)化简一下,得:\[i\timesA_1+(i-1)\timesA_2+\dots+1\timesA_x\]在\(a\)每增加\(1\)时,这个和\(s\)将会变......
  • [题解]AT_abc250_d [ABC250D] 250-like Number
    思路对于这道题,我们可以发现一个事情:我们筛质数只需要筛\(1\sim\log_3n\)的部分就行了。因为\(k=p\timesq^3\),那么,我们考虑一种极端情况,\(p\)为一个很小的数,那么\(k\)就无限接近于\(q^3\)。我们就先假设\(k=q^3\),那么可以得出\(q=\log_3n\)。然后由题目描......
  • [题解]AT_abc249_f [ABC249F] Ignore Operations
    思路反悔贪心套路题。发现一个性质,当一个操作1生效意味着在这一步之前的所有操作都没用。那么考虑倒着枚举,对于每一个操作1的选取状态做一个简单的分讨:如果保留,那么这种情况下的答案就是之前的\(sum\)加上当前的\(y\)。如果不保留,继续往前走,\(k\leftarrowk-1\)。......
  • [题解]AT_abc249_e [ABC249E] RLE
    思路定义\(dp_{i,j}\)表示在前\(i\)位原字符串,压缩为\(j\)位的方案数。不难得出状态转移方程:\[dp_{i,j}=\sum_{k=1}^{i}(25\timesdp_{k,j-\lfloor\log_{10}k\rfloor-1})\]这样搞是\(\Theta(n^3)\)的,所以考虑优化。不难发现\(\log_{10}k\)是......
  • [题解]AT_abc248_e [ABC248E] K-colinear Line
    思路首先,我们得清楚如何判断三点共线。对于每一个点,它的横纵坐标都有这么一个关系:\(n\timesx+m=y\)(其中\(n,m\)为常数)。那么,对于三点共线的点来说,\(n,m\)是相同的。因此我们得出三个式子。\[n\timesx_a+m=y_a\]\[n\timesx_b+m=y_b\]\[n\tim......
  • [题解]AT_abc248_d [ABC248D] Range Count Query
    思路其实很简单,我们可以将所有数值相同的值的下标存入一个vector里面。因为,我们既然要查找\(X\),不妨把所有值为\(X\)的下标存在一起,方便查找。(可以在输入的时候完成)我们不妨在每一个数值后面添加一个哨兵,然后二分查找第一个大于等于\(l\)的数和第一个大于等于\(r+1\)......
  • [题解]AT_abc253_g [ABC253G] Swap Many Times
    思路首先,不难看出一个规律,就是对于一个序列\(a\),如果它将操作所有以\(x\)为第一关键字的二元组,那么序列的\(a_{x\simn}\)将循环右移一位。(注意,在这里的\(x\)指的是在\(1\sim(n-1)\)中的任意一个定值)那么,我们就可以将编号分别为\(l\simr\)的这些二元组分为三......
  • [题解]AT_abc250_e [ABC250E] Prefix Equality
    思路对于这种题目,通常会想到用哈希维护。由于集合相同与\(a_{1\simx}\),\(b_{1\simy}\)的顺序无关,所以对于我们的哈希函数\(h(x)\)必定需要用一种有交换律的符号。首先想到的当然是加法,但是不太好实现,因为这些数太大了,不因会爆unsignedlonglong,还会爆__int128,所以不......