结论:排列$p'_{i}$可以通过排列$p_{i}$得到当且仅当$\forall 1\le i<j<i+k,(p_{i}-p_{j})(p'_{i}-p'_{j})>0$

证明:构造$b_{p_{i}}=i$,交换即令$b_{i}$与$b_{i+1}$交换,条件为$|b_{i}-b_{i+1}|\ge k$,那么对于$x<y$的$|b_{x}-b_{y}|<k$,相对位置不会发生改变,同时其他位置可以任意交换,反映在$p'_{i}$上即$p'_{b_{x}}$和$p'_{b_{y}}$的相对大小不会改变,即得证

考虑从$n$到$1$依次填数,统计每一个位置上$cnt_{i}=\sum_{j=\max(i-k+1,1)}^{\min(i+k-1,n)}[p_{i}<p_{j}]$,然后不断找到最后一个$cnt_{i}$为0且未被填过的位置填$i$,并将其左右长度为$2k-1$的区间的$cnt_{i}$减1

先考虑贪心填$n$的正确性,由于这些位置之间的距离必然大于$k$(否则不可能同时$cnt_{i}=0$),那么当$n$填在了不是最后一个的位置,两者交换必然更优

之后的问题相当于当前问题的子问题,已经填过的位置可以看作最小的(因为已经对之前的-1),那么新的最大值$n-1,n-2,...$的填法与$n$相同

那么用线段树来维护,相当于支持:1.区间-1;2.查询最后一个0,为了方便查询,可以将填写后的$cnt_{i}$置为$\infty

 1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 500005
4 #define L (k<<1)
5 #define R (L+1)
6 #define mid (l+r>>1)
7 int n,k,x,a[N],cnt[N],ans[N],f[N<<3],tag[N<<3];
8 void upd(int k,int x){
9 tag[k]+=x;
10 f[k]+=x;
11 }
12 void down(int k){
13 upd(L,tag[k]);
14 upd(R,tag[k]);
15 tag[k]=0;
16 }
17 void update(int k,int l,int r,int x){
18 if (l==r){
19 f[k]=1;
20 return;
21 }
22 if (x<=mid)update(L,l,mid,x);
23 else update(R,mid+1,r,x);
24 f[k]=f[L]+f[R];
25 }
26 int query(int k,int l,int r,int x,int y){
27 if ((l>y)||(x>r))return 0;
28 if ((x<=l)&&(r<=y))return f[k];
29 return query(L,l,mid,x,y)+query(R,mid+1,r,x,y);
30 }
31 void update(int k,int l,int r,int x,int y,int z){
32 if ((l>y)||(x>r))return;
33 if ((x<=l)&&(r<=y)){
34 upd(k,z);
35 return;
36 }
37 down(k);
38 update(L,l,mid,x,y,z);
39 update(R,mid+1,r,x,y,z);
40 f[k]=min(f[L],f[R]);
41 }
42 int find(int k,int l,int r){
43 if (l==r)return l;
44 down(k);
45 if (!f[R])return find(R,mid+1,r);
46 return find(L,l,mid);
47 }
48 int main(){
49 scanf("%d%d",&n,&k);
50 for(int i=1;i<=n;i++){
51 scanf("%d",&x);
52 a[x]=i;
53 }
54 for(int i=n;i;i--){
55 cnt[i]=query(1,1,n,max(a[i]-k+1,1),min(a[i]+k-1,n));
56 update(1,1,n,a[i]);
57 }
58 memset(f,0,sizeof(f));
59 for(int i=1;i<=n;i++)update(1,1,n,a[i],a[i],cnt[i]);
60 for(int i=n;i;i--){
61 int l=find(1,1,n);
62 ans[l]=i;
63 update(1,1,n,l,l,0x3f3f3f3f);
64 update(1,1,n,max(l-k+1,1),min(l+k-1,n),-1);
65 }
66 for(int i=1;i<=n;i++)printf("%d\n",ans[i]);
67 }

[atAGC001F]Wide Swap的更多相关文章

  1. 【AtCoder Grand Contest 001F】Wide Swap [线段树][拓扑]

    Wide Swap Time Limit: 50 Sec  Memory Limit: 512 MB Description Input Output Sample Input 8 3 4 5 7 8 ...

  2. AT1984 Wide Swap

    AT1984 Wide Swap 题意翻译 给出一个元素集合为\(\{1,2,\dots,N\}(1\leq N\leq 500,000)\)的排列\(P\),当有\(i,j(1\leq i<j ...

  3. AtCoder AGC001F Wide Swap (线段树、拓扑排序)

    题目链接: https://atcoder.jp/contests/agc001/tasks/agc001_f 题解: 先变成排列的逆,要求\(1\)的位置最小,其次\(2\)的位置最小,依次排下去( ...

  4. AGC001F - Wide Swap

    Description 给你一个长度为$n$的排列,每次可以交换$|i-j|\geq K$并且$|a_i-a_j|=1$的数对,问你经过若干次变换后最小字典序的排列是啥 Solution 对$a$做一 ...

  5. AGC001 F - Wide Swap【线段树+堆+拓扑排序】

    给出的模型很难搞,所以转换一下,记p[i]为i这个数的位置,然后相邻两个p值差>k的能交换,发现使原问题字典序最小也需要使这里的字典序最小 注意到p值差<=k的前后顺序一定不変,那么可以n ...

  6. AtCoder Grand Contest 001F Wide Swap

    解法参考这位大佬的:https://www.cnblogs.com/BearChild/p/7895719.html 因为原来的数组不好做于是我们想反过来数组,根据交换条件:值相邻且位置差大于等于k, ...

  7. 题解 Wide Swap

    题目传送门 题目大意 给出一个长度为 \(n\) 的排列 \(a_{1,2,...,n}\) 以及常数 \(k\),每次可以交换两个数 \(a_i,a_j\) 当且仅当 \(j-i\ge k \tex ...

  8. Atcoder Grand Contest 001 F - Wide Swap(拓扑排序)

    Atcoder 题面传送门 & 洛谷题面传送门 咦?鸽子 tzc 来补题解了?奇迹奇迹( 首先考虑什么样的排列可以得到.我们考虑 \(p\) 的逆排列 \(q\),那么每次操作的过程从逆排列的 ...

  9. RE:从零开始的AGC被虐(到)生活(不能自理)

    RE:从零开始的AGC被虐(到)生活(不能自理) 「一直注视着你,似近似远,总是触碰不到.」 --来自风平浪静的明天 AtCoder Grand Contest 001 B: Mysterious L ...

随机推荐

  1. The type name or alias SqlServer could not be resolved.Please check your configuration

    The type name or alias SqlServer could not be resolved.Please check your configuration file.... 检查一下 ...

  2. Data Management Tools(数据管理工具)《二》

    (数据管理工具)<二> 点击跳转(数据管理工具)<一> 16.打包 # Process: 共享包 arcpy.SharePackage_management("&qu ...

  3. allure报告中allure.title 如何去掉后方的参数化显示

    1.解决方法如下 listener.py 文件位置:Lib\site-packages\allure_pytest\listener.py (第三方包所在的LIb目录) 将下图中红色部分test_re ...

  4. RA-28000 账号被锁定的解决办法

    ORA-28000 账号被锁定的解决办法 错误场景:当使用sqlplus进行登录时报错:ORA-28000 账号被锁定.错误原因:由于oracle 11g 在默认在default概要文件中设置了密码最 ...

  5. 写了10000条Airtest截图脚本总结出来的截图经验,赶紧收藏!

    前言 今天想先给大家分享1个小白用户的Airtest从入门到放弃的故事: 小A是一个自动化的小白,在逛测试论坛的时候,偶然间发现了Airtest这个基于图像识别的UI自动化框架. 出于好奇,小A试用了 ...

  6. BUAA 2020 软件工程 提问回顾与个人总结

    BUAA 2020 软件工程 提问回顾与个人总结 Author: 17373051 郭骏 项目 内容 这个作业属于哪个课程 2020春季计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 提问回顾 ...

  7. Java 将Word保存为WPS和WPT格式

    本文通过Java示例展示将Word文档(如.doc/.docx)保存为WPS和WPT格式的方法. 程序环境配置 IntelliJ IDEA 2018(jdk 1.8.0) Word Jar包:Spir ...

  8. 『学了就忘』Linux基础 — 13、Linux系统的分区和格式化

    目录 1.Linux系统的分区 (1)磁盘分区定义 (2)两种分区表形式 (3)MBR分区类型 2.Linux系统的格式化 (1)格式化定义 (2)格式化说明 1.Linux系统的分区 (1)磁盘分区 ...

  9. triangle leetcode C++

    Given a triangle, find the minimum path sum from top to bottom. Each step you may move to adjacent n ...

  10. hdu 1160 FatMouse's Speed(最长不下降子序列+输出路径)

    题意: FatMouse believes that the fatter a mouse is, the faster it runs. To disprove this, you want to ...