首页 > 其他分享 >[题解]AT_abc253_g [ABC253G] Swap Many Times

[题解]AT_abc253_g [ABC253G] Swap Many Times

时间:2024-06-22 10:59:40浏览次数:12  
标签:右移 二元 int 题解 复杂度 ABC253G Times 序列 sim

思路

首先,不难看出一个规律,就是对于一个序列 \(a\),如果它将操作所有以 \(x\) 为第一关键字的二元组,那么序列的 \(a_{x \sim n}\) 将循环右移一位。(注意,在这里的 \(x\) 指的是在 \(1 \sim (n - 1)\) 中的任意一个定值)

那么,我们就可以将编号分别为 \(l \sim r\) 的这些二元组分为三组:

  1. \((x_1,y_1) \sim (x_1,n)\),其中 \((x_1,y_1)\) 为编号为 \(l\) 的二元组。
  2. \((x_1 + 1,x_1 + 2) \sim (x_2,n)\),其中 \((x_2,n)\) 为编号不大于 \(r\) 的最后一个完整操作区间的最后一个二元组。(完整操作区间表示对于一个 \(x\),\((x,x + 1) \sim (x,n)\) 都会被取到的二元组区间)
  3. \((x_2 + 1,x_2 + 2) \sim (x_2 + 1,y_2)\),其中 \((x_2 + 1,y_2)\) 为编号为 \(r\) 的二元组。

然后分别维护这三种情况即可:

  1. 首先,定义 \(sum_i = \sum_{j = 1}^{n - i}(n - j)\),那么,我们可以二分得出二元组 \((x_1,y_1)\),然后暴力维护 \((x_1,y_1) \sim (x_1,n)\) 即可。时间复杂度 \(\Theta(n)\)。
  2. 由上文的规律,我们只需要将每种状态循环右移一位即可,时间复杂度 \(\Theta(n \sqrt n)\),考虑优化。在这里先举一个例子,那么我们可以用两个 vector \(A,B\) 维护此过程(其中 \(A\) 表示循环右移时最后的元素走到序列前面的元素,\(B\) 表示循环右移时没有走到序列前面的元素),对于每一次循环右移,都会将 \(B\) 中的最后一个元素放在 \(A\) 的末尾,时间复杂度 \(\Theta(\sqrt n)\)。然后将 \(A,B\) 拼起来得到当前序列 \(a\)。需要注意的是,\(B\) 的初始状态是将 \(x_1 \sim n\) 放入,因为在 \(x_1\) 之前的根本不会动。
  3. 将剩下的操作次数全部花光即可。

Code

#include <bits/stdc++.h>  
#define int long long  
#define re register  
  
using namespace std;  
  
const int N = 2e5 + 10;  
int n,l,r;  
int sum[N],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 << 1) + (r << 3) + (c ^ 48);  
        c = getchar();  
    }  
    return r * w;  
}  
  
inline void init(){//预处理 sum 数组   
    for (re int i = 1,k = n - 1;i < n;i++,k--) sum[i] = sum[i - 1] + k;  
}  
  
signed main(){  
    n = read();  
    l = read();  
    r = read();  
    init();  
    for (re int i = 1;i <= n;i++) arr[i] = i;  
    int a = lower_bound(sum + 1,sum + n/*注意这里只能不能写 + 1,因为 sum[n] = 0,加上后 sum 数组不有序无法二分*/,l) - sum;//找出 (x1,y1)   
    int b = a + l - sum[a - 1];  
    int cnt = 1;  
    while (cnt <= (r - l + 1) && b <= n){//暴力维护 (x1,y1) ~ (x1,n)   
        swap(arr[a],arr[b]);  
        cnt++;  
        b++;  
    }  
    int i = a + 1;  
    vector<int> A,B;//维护中间完整段   
    for (re int j = i;j <= n;j++) B.push_back(arr[j]);  
    while (sum[i] <= r && i < n){  
        if (B.empty()) break;  
        A.push_back(B.back());  
        B.pop_back();  
        i++;  
    }  
    for (auto x:A) arr[++a] = x;//更新新的序列   
    for (auto x:B) arr[++a] = x;  
    cnt = r - sum[i - 1];  
    for (int j = 1,k = i + 1;j <= cnt && k <= n;j++,k++) swap(arr[i],arr[k]);//暴力维护剩余的操作次数   
    for (re int i = 1;i <= n;i++) printf("%lld ",arr[i]);  
    return 0;  
}  

标签:右移,二元,int,题解,复杂度,ABC253G,Times,序列,sim
From: https://www.cnblogs.com/WaterSun/p/18261977

相关文章

  • [题解]AT_abc250_e [ABC250E] Prefix Equality
    思路对于这种题目,通常会想到用哈希维护。由于集合相同与\(a_{1\simx}\),\(b_{1\simy}\)的顺序无关,所以对于我们的哈希函数\(h(x)\)必定需要用一种有交换律的符号。首先想到的当然是加法,但是不太好实现,因为这些数太大了,不因会爆unsignedlonglong,还会爆__int128,所以不......
  • [题解]AT_abc256_g [ABC256G] Black and White Stones
    思路容易看出来是个DP题,但是你发现DP的起点是不好确定的,于是假定第一条边的起点是黑色。然后你发现设为白色的贡献与黑色是相同的,于是直接令第一条边的起点是黑色,最后答案乘以\(2\)即可。然后就可以愉快的DP了。首先枚举每条边白色点的数量\(k\),定义\(dp_{i,0/1}\)......
  • [题解]AT_abc225_f [ABC225F] String Cards
    思路Part1弱化版看到这道题的第一眼想到了P1012这道题。但是,这两道题选择的数量是有区别的。我们可以由拼数得出一个结论性的排序规则(这里就不多做解释了):inlineboolcmp(stringa,stringb){returna+b<b+a;}如果用这样的做法,有hack。Part2状态......
  • [题解]AT_abc255_d [ABC255D] ±1 Operation 2
    思路因为\(1\leqn,q\leq2\times10^5\),所以对于每一次查询的时间复杂度一定要达到\(\Theta(\logn)\),甚至于\(\Theta(1)\)。一个最简单的想法,我们先统计出整个序列\(a\)的和\(sum\),然后答案是\(|sum-x\timesn|\)。很显然,这个想法是错误的,因为对于\(a\)中只有......
  • [题解]AT_abc225_e [ABC225E] フ
    思路对于每一个7,我们都可以抽象为这样一个图形:如果有两个7,无论它是否有重合部分,红色部分是不需要判断的,只需要看绿色的部分。因此,我们的问题就简化为了三角形,而不是四边形。对于所有的7,都有一个公共顶点:\((0,0)\)点。所以,我们可以引出一个叫斜率的概念来判断这些三角形......
  • [题解]AT_abc225_d [ABC225D] Play Train
    题意给定\(N\)个小车,每个小车的编号分别为:\(1,2,\dots,N\)。现在有\(Q\)个操作,每个操作执行\(3\)种操作:1xy,将\(x\)和\(y\)相连。(\(y\)在\(x\)之后)2xy,将\(x\)和\(y\)的连接解除。3x,输出\(x\)所在链的长度,及其这条链中的所有元素。(从前往后)思路我......
  • [题解]AT_abc224_e [ABC224E] Integers on Grid
    比较符合CCF造数据水平的题。思路首先可以用两个vector<pair<int,int>>v[N]分别将每一行、每一列的元素的权值与编号存储下来。那么可以对所有的\(v_i\)按照权值从小到大排序。那么发现对于所有的满足v[i][p].fst<v[i][q].fst的\((p,q)\)都可以建一条从\(p\)指......
  • [题解]AT_abc222_f [ABC222F] Expensive Expense
    板子题,模拟赛场切了。思路线段树换根板子题。因为需要求每一个点的答案,所以定义\(dp_i\)表示以\(i\)为根的最长距离。考虑将一个点\(v\)转化为根,树的形态会发生什么变化(假设\(v\)的父亲节点是\(u\))。发现在\(v\)子树中的节点,距离都会减少\(w_{u\tov}\),其它节点......
  • [题解]AT_abc217_g [ABC217G] Groups
    思路定义\(dp_{i,j}\)表示将前\(i\)个数,正好分为\(j\)组的方案数。那么,我们对\(i\)号元素进行分类讨论:将\(i\)放入原本就存在的组中,因为在同一个组中不能存在两个数\(x,y\),使得\(x\bmodm=y\bmodm\)。所以对于\(i\),如果它是\(m\)的倍数,则在\(1\simi-......
  • [题解]AT_abc217_f [ABC217F] Make Pair
    思路区间DP好题,合并的时候十分毒瘤。首先,定义\(dp_{i,j}\)表示合并\([i,j]\)区间不同的方案的数量。不难发现,如果区间长度为奇数(即\(j-i+1\)为奇数),一定无法合并。然后,如果\(i,j\)是朋友关系,有\(dp_{i,j}=dp_{i+1,j-1}\)。接着,我们可以枚举一个中间点\(......